From e8815c1d95afe888e9e0797c462e0da1963d2d25 Mon Sep 17 00:00:00 2001 From: Sumner Date: Wed, 17 Jul 2019 14:59:28 +0200 Subject: [PATCH 1/2] major refractor --- build/js/d3sm.min.v0.0.3.js | 2 - build/js/d3sm.min.v0.0.4.js | 2 - build/js/d3sm.v0.0.3.js | 10613 --- build/js/d3sm.v0.0.4.js | 10613 --- d3sm-legacy.js | 2171 - data/1-col-heatmap.js | 1138 - data/2-col-heatmap.js | 6924 -- data/d3sm-data.js | 933 - data/d3sm-logo.afdesign | Bin 47050 -> 0 bytes data/d3sm.icon.png | Bin 2013 -> 0 bytes data/d3sm.png | Bin 73103 -> 0 bytes data/d3sm.svg | 1 - data/data.js | 13 - data/datasets.js | 135 - data/icon.afdesign | Bin 23157 -> 0 bytes data/upset.js | 72292 ---------------- data/upset0.js | 5583 -- data/upset1.js | 23910 ----- demos/axes/index.html | 279 - demos/axes/js/v001.js | 196 - demos/axes/js/v003.js | 0 .../index.html | 195 - .../js/v001.js | 125 - demos/basic-violins/index.html | 172 - demos/box-whiskers-points/index.html | 154 - demos/box-whiskers/index.html | 155 - demos/bubble-heatmap/index.html | 161 - demos/bubble-heatmap/js/v000.js | 96 - demos/bundle/index.html | 52 - demos/filter-select/index.html | 155 - demos/filter-select/js/v002.txt | 0 demos/filter-table/index.html | 120 - demos/filter-table/js/v003.js | 222 - demos/heatmap/index.html | 70 - demos/heatmap/js/v000.js | 98 - demos/heatmap/js/v003.js | 132 - demos/index.html | 37 +- demos/scatter/index.html | 141 - demos/scatter/js/v001.js | 103 - demos/scatter/js/v002.js | 117 - demos/scatter/js/v003.js | 187 - demos/tooltip-design/index.html | 86 - demos/upset/index.html | 131 - demos/upset/js/v003.js | 300 - demos/vertical-violins/index.html | 147 - dist/d3sm.esm.js | 10601 +-- dist/d3sm.min.js | 2 +- dist/d3sm.umd.js | 10601 +-- index.html | 6 +- needs-to-be-bundled/distribution.js | 241 - needs-to-be-bundled/graph.js | 806 - needs-to-be-bundled/scatter.js | 265 - needs-to-be-bundled/tooltip.js | 79 - rollup.config.js | 34 +- src/entry.js | 27 + src/modules/aux/index.js | 9 + src/{scripts/modules => modules/aux}/lasso.js | 64 +- .../aux}/multi-plot-zoom.js | 29 +- .../modules => modules/aux}/plot-zoom.js | 29 +- src/{scripts => }/modules/axis.js | 77 +- .../modules => modules/charts}/bar.js | 22 +- .../modules => modules/charts}/box-whisker.js | 38 +- .../charts}/bubble-heatmap.js | 60 +- .../modules => modules/charts}/heatmap.js | 54 +- src/modules/charts/index.js | 14 + .../modules => modules/charts}/scatter.js | 13 +- .../modules => modules/charts}/upset.js | 46 +- .../modules => modules/charts}/violin.js | 83 +- src/{scripts => }/modules/color-function.js | 6 +- src/{scripts => }/modules/d3-prototypes.js | 4 +- src/{scripts => }/modules/grouping-spacer.js | 10 +- .../legends}/categorical-legend.js | 23 +- src/modules/legends/index.js | 9 + .../legends}/numeric-legend.js | 38 +- src/{scripts => }/modules/tooltip.js | 31 +- .../utils/array.js} | 17 +- src/modules/utils/colors.js | 32 + src/modules/utils/console.js | 97 + src/modules/utils/index.js | 16 + src/modules/utils/math.js | 160 + src/modules/utils/misc.js | 139 + src/modules/utils/paths.js | 40 + src/modules/utils/selections.js | 120 + src/modules/utils/strings.js | 41 + src/scripts/main.js | 116 - src/scripts/modules/data-toggle.js | 210 - src/scripts/modules/filter-table.js | 118 - src/scripts/modules/helpers.js | 273 - src/scripts/modules/lasso-leg.js | 748 - src/scripts/modules/lasso-widget.js | 606 - src/scripts/modules/points.js | 95 - src/scripts/modules/select-filter.js | 110 - src/scripts/modules/utils.js | 599 - src/styles/main.css | 3 - 94 files changed, 9872 insertions(+), 154950 deletions(-) delete mode 100644 build/js/d3sm.min.v0.0.3.js delete mode 100644 build/js/d3sm.min.v0.0.4.js delete mode 100644 build/js/d3sm.v0.0.3.js delete mode 100644 build/js/d3sm.v0.0.4.js delete mode 100644 d3sm-legacy.js delete mode 100644 data/1-col-heatmap.js delete mode 100644 data/2-col-heatmap.js delete mode 100644 data/d3sm-data.js delete mode 100644 data/d3sm-logo.afdesign delete mode 100644 data/d3sm.icon.png delete mode 100644 data/d3sm.png delete mode 100644 data/d3sm.svg delete mode 100644 data/data.js delete mode 100644 data/datasets.js delete mode 100644 data/icon.afdesign delete mode 100644 data/upset.js delete mode 100644 data/upset0.js delete mode 100644 data/upset1.js delete mode 100644 demos/axes/index.html delete mode 100644 demos/axes/js/v001.js delete mode 100644 demos/axes/js/v003.js delete mode 100644 demos/bar-chart-same-data-complex-grouping/index.html delete mode 100644 demos/bar-chart-same-data-complex-grouping/js/v001.js delete mode 100644 demos/basic-violins/index.html delete mode 100644 demos/box-whiskers-points/index.html delete mode 100644 demos/box-whiskers/index.html delete mode 100644 demos/bubble-heatmap/index.html delete mode 100644 demos/bubble-heatmap/js/v000.js delete mode 100644 demos/bundle/index.html delete mode 100644 demos/filter-select/index.html delete mode 100644 demos/filter-select/js/v002.txt delete mode 100644 demos/filter-table/index.html delete mode 100644 demos/filter-table/js/v003.js delete mode 100644 demos/heatmap/index.html delete mode 100644 demos/heatmap/js/v000.js delete mode 100644 demos/heatmap/js/v003.js delete mode 100644 demos/scatter/index.html delete mode 100644 demos/scatter/js/v001.js delete mode 100644 demos/scatter/js/v002.js delete mode 100644 demos/scatter/js/v003.js delete mode 100644 demos/tooltip-design/index.html delete mode 100644 demos/upset/index.html delete mode 100644 demos/upset/js/v003.js delete mode 100644 demos/vertical-violins/index.html delete mode 100644 needs-to-be-bundled/distribution.js delete mode 100644 needs-to-be-bundled/graph.js delete mode 100644 needs-to-be-bundled/scatter.js delete mode 100644 needs-to-be-bundled/tooltip.js create mode 100644 src/entry.js create mode 100644 src/modules/aux/index.js rename src/{scripts/modules => modules/aux}/lasso.js (87%) rename src/{scripts/modules => modules/aux}/multi-plot-zoom.js (88%) rename src/{scripts/modules => modules/aux}/plot-zoom.js (87%) rename src/{scripts => }/modules/axis.js (91%) rename src/{scripts/modules => modules/charts}/bar.js (96%) rename src/{scripts/modules => modules/charts}/box-whisker.js (94%) rename src/{scripts/modules => modules/charts}/bubble-heatmap.js (91%) rename src/{scripts/modules => modules/charts}/heatmap.js (92%) create mode 100644 src/modules/charts/index.js rename src/{scripts/modules => modules/charts}/scatter.js (97%) rename src/{scripts/modules => modules/charts}/upset.js (84%) rename src/{scripts/modules => modules/charts}/violin.js (92%) rename src/{scripts => }/modules/color-function.js (98%) rename src/{scripts => }/modules/d3-prototypes.js (95%) rename src/{scripts => }/modules/grouping-spacer.js (97%) rename src/{scripts/modules => modules/legends}/categorical-legend.js (93%) create mode 100644 src/modules/legends/index.js rename src/{scripts/modules => modules/legends}/numeric-legend.js (75%) rename src/{scripts => }/modules/tooltip.js (85%) rename src/{scripts/modules/array-functions.js => modules/utils/array.js} (92%) create mode 100644 src/modules/utils/colors.js create mode 100644 src/modules/utils/console.js create mode 100644 src/modules/utils/index.js create mode 100644 src/modules/utils/math.js create mode 100644 src/modules/utils/misc.js create mode 100644 src/modules/utils/paths.js create mode 100644 src/modules/utils/selections.js create mode 100644 src/modules/utils/strings.js delete mode 100644 src/scripts/main.js delete mode 100644 src/scripts/modules/data-toggle.js delete mode 100644 src/scripts/modules/filter-table.js delete mode 100644 src/scripts/modules/helpers.js delete mode 100644 src/scripts/modules/lasso-leg.js delete mode 100644 src/scripts/modules/lasso-widget.js delete mode 100644 src/scripts/modules/points.js delete mode 100644 src/scripts/modules/select-filter.js delete mode 100644 src/scripts/modules/utils.js delete mode 100644 src/styles/main.css diff --git a/build/js/d3sm.min.v0.0.3.js b/build/js/d3sm.min.v0.0.3.js deleted file mode 100644 index fcd6774..0000000 --- a/build/js/d3sm.min.v0.0.3.js +++ /dev/null @@ -1,2 +0,0 @@ -var d3sm=function(t){"use strict";function e(t,e,n){return n.indexOf(t)===e}function n(t){var e=document.createElementNS("http://www.w3.org/2000/svg","g");t=void 0==t?"translate(0,0)":t,e.setAttributeNS(null,"transform",t);var n=e.transform.baseVal.consolidate().matrix;return[n.e,n.f]}function r(t,e){(t=String(t).replace(/[^0-9a-f]/gi,"")).length<6&&(t=t[0]+t[0]+t[1]+t[1]+t[2]+t[2]),e=e||0;var n,r,o="#";for(r=0;r<3;r++)n=parseInt(t.substr(2*r,2),16),o+=("00"+(n=Math.round(Math.min(Math.max(0,n+n*e),255)).toString(16))).substr(n.length);return o}function o(t,e){var n=d3.median(t),r=t.filter(function(t){return tn}),a=void 0==(a=d3.median(r))?n:a,i=void 0==(i=d3.min(r))?a:i,l=void 0==(l=d3.median(o))?n:l,c=void 0==(c=d3.max(o))?l:c,u="q0",s="q1",d="q2",f="q3",h="q4",g={};return void 0!=e&&5==e.length&&(u=e[0],s=e[1],d=e[2],f=e[3],h=e[4]),g[u]=i,g[s]=a,g[d]=n,g[f]=l,g[h]=c,g}function a(){return Array.prototype.slice.call(arguments).join("-")}function i(t,e){var n=function(t,e,n){n&&(e=-e);var r=(""+t).split("e");return+(r[0]+"e"+(r[1]?+r[1]+e:e))};return n(Math.round(n(t,e,!1)),e,!0)}function l(t){var e=t.parentElement,n=e.tagName.toLowerCase();return"svg"===n?e:"html"!==n?l(e):void 0}function c(t,e,n){var r=void 0==n?"":"."+n,o=t.select(e+r).empty()?t.append(e):t.select(e+r);return o.classed(r.replace(".",""),!0).attr("transform",void 0==o.attr("transform")?"translate(0,0)":o.attr("transform"))}function u(t,e,n){for(var r=[t],o=(e-t)/(n-1),a=0;a1?o:t*o),l=(i=i<0?0:i)/e;return a&&void 0!=n&&l=r.length?r.push(e.length-1):r[n]+=e.length-1;e.map(function(e,o){Array.isArray(e)&&t(e,n,r)});return r}(t);o=(e-n*r)/i.map(function(t,e){return 1*t/(e+1)}).reduce(function(t,e){return t+e},0);return isNaN(o)?0:o}function b(t,e,n,r,o,a,i){if("up"!=t&&"top"!=t&&1!=t||(t=!0),"down"!=t&&"bottom"!=t&&0!=t||(t=!1),i=void 0==i?"horizontal":i,a=void 0==a?1:a,"horizontal"!=i){var l=o*a,c=(r=t?r:-r,t?e+r:e),u=t?e:e+r,s=t?c:u;return f="M "+c+" "+o/2+" L "+u+" "+o/2+" M "+s+" "+(o/2-l/2)+" L "+s+" "+(o/2+l/2)+" "}var d=r*a,f="M "+r/2+" "+(c=t?n+o:n)+" L "+r/2+" "+(u=t?n:n+o)+" h "+-d/2+" 0 h "+d+" 0 ";return f}function k(){var t,e,n,r=!0,o=d3.scaleLinear(),a="category",i="d3sm-groupped-item",l=1e3,c=d3.easeSin,u="spacer",s=function(t){t.attr("transform",function(t,e){return"translate("+(r?window.outerWidth:0)+","+(r?0:window.outerWidth)+")"})},d=function(t){m("groupingSpacer","exiting with",{current:t,currentNode:t.node()}),t.selectAll("g").classed("to-remove",!0),t.transition().duration(.9*l).ease(c).attr("transform",function(t,e){return"translate("+(r?window.outerWidth:0)+","+(r?0:window.outerWidth)+")"}).remove()};function f(t,h,g){void 0==g&&(g=0);var p=t.selectAll("g."+u+'[level="'+g+'"]').data(h),m=p.enter().append("g").attr("level",g).attr("class",u),v=p.exit();p=p.merge(m),"function"==typeof d?v.each(function(t,e){d(d3.select(this))}):v.remove();var y=n/(g+1),x=0;return p.each(function(t,n){var h=d3.select(this);if(void 0==h.attr("transform")&&"function"==typeof s&&s(h),h.transition().duration(l).ease(c).attr("transform",function(t,e){return"translate("+(r?"scale"==a?o(t):x:0)+","+(r?0:"scale"==a?o(t):x)+")"}),Array.isArray(t)){x+=f(h,t,g+1);var m=h.selectAll("g."+u+'[level="'+g+'"] > g.'+i+"."+u);"function"==typeof d?m.each(function(t,e){d(d3.select(this))}):m.remove()}else{x+=e;var v=h.select("g."+u+'[level="'+g+'"] > g.'+i+"."+u);v.empty()&&(v=h.append("g").attr("class",i).classed(u,!0)),v.attr("parent-index",n);m=h.selectAll("g."+u+'[level="'+(g+1)+'"]');"function"==typeof d?m.each(function(t,e){d(d3.select(this))}):m.remove()}x+=n==p.size()-1?0:y}),x}return f.horizontalQ=function(t){return arguments.length?(r=t,f):r},f.scale=function(t){return arguments.length?(o=t,f):o},f.moveby=function(t){return arguments.length?(a=t,f):a},f.numberOfObjects=function(e){return arguments.length?(t=e,f):t},f.objectClass=function(t){return arguments.length?(i=t,f):i},f.objectSize=function(t){return arguments.length?(e=t,f):e},f.spacerSize=function(t){return arguments.length?(n=t,f):n},f.transitionDuration=function(t){return arguments.length?(l=t,f):l},f.easeFunc=function(t){return arguments.length?(c=t,f):c},f.namespace=function(t){return arguments.length?(u=t,f):u},f.enterFunction=function(t){return arguments.length?(s=t,f):s},f.exitFunction=function(t){return arguments.length?(d=t,f):d},f}var w=function(){return function(t,e){if(Array.isArray(t))return t;if(Symbol.iterator in Object(t))return function(t,e){var n=[],r=!0,o=!1,a=void 0;try{for(var i,l=t[Symbol.iterator]();!(r=(i=l.next()).done)&&(n.push(i.value),!e||n.length!==e);r=!0);}catch(t){o=!0,a=t}finally{try{!r&&l.return&&l.return()}finally{if(o)throw a}}return n}(t,e);throw new TypeError("Invalid attempt to destructure non-iterable instance")}}(),S=function(t){if(Array.isArray(t)){for(var e=0,n=Array(t.length);ewindow.innerWidth-window.scrollX&&(d=d3.event.pageX-b.width-15),f+b.height>window.innerHeight-window.scrollY&&(f=d3.event.pageY-b.height-15),"relative"==h.style("position")?h.style("position","absolute").style("left",d+"px").style("top",f+"px"):h.style("left",d+"px").style("top",f+"px"),h.attr("z-index",1e4)}return a.keys=function(t){return arguments.length?(e=t,a):e},a.values=function(t){return arguments.length?(n=t,a):n},a.header=function(t){return arguments.length?(r=t,a):r},a.data=function(t){return arguments.length?(o=t,a):o},a.selection=function(e){return arguments.length?(t=e,a):t},a}function A(t){var e,n="d3sm-select-filter",r="Select options:",o=void 0,i=void 0;function l(){var o=c(t,"div","input-group").classed(a(n,"container"),!0),l=(c(c(o,"div","select-prepend").classed("input-group-prepend",!0),"span","input-group-text").text(r),c(o,"select","custom-select").classed(a(n,"select"),!0)),s=c(c(o,"div","select-append").classed("input-group-prepend",!0),"a","filter-button").classed("btn btn-outline-secondary",!0),d=(c(s,"i","fa fa-filter"),c(o,"div","filter-input-group").classed("input-group",!0).classed("d-none",!0)),f=(c(c(c(d,"div","input-group-prepend"),"span","input-group-text").classed("search-button",!0),"i","fa fa-search"),c(d,"input","form-control").attr("placeholder","all").attr("type","text")),h=c(c(d,"div","input-group-append"),"a","close-button").classed("btn btn-outline-secondary",!0),g=(c(h,"i","fa fa-close"),d3.keys(e)),p=l.selectAll("option");p=(p=p.data(d3.keys(e))).merge(p.enter().append("option")).attr("value",function(t,e){return t}).text(function(t,e){return t});var m=h;s.on("click",function(t,e){var n=d.classed("d-none");d.classed("d-none",!n)}),m.on("click",function(t,e){f.property("value","").dispatch("input")}),f.on("input",function(t,n){var r,o=f.property("value"),a=new RegExp(o,"gi");""==o?r=g:(r=[],d3.keys(e).map(function(t,e){var n=t.match(a);null==n||""==n.join("")||r.push(t)})),(p=(p=l.selectAll("option")).data(r)).exit().remove(),p=p.merge(p.enter().append("option")).attr("value",function(t,e){return t}).text(function(t,e){return t});var c=u();i!=c&&(i=c,l.dispatch("change"))})}function u(){var n=t.select("select").property("value");return void 0==n||""==n?void 0==o?d3.keys(e)[0]:o:n}return l.data=function(t){return arguments.length?(e=t,l):e},l.namespace=function(t){return arguments.length?(n=t,l):n},l.selectionName=function(t){return arguments.length?(r=t,l):r},l.defaultValue=function(t){return arguments.length?(o=t,l):o},l.currentOption=u,l}function M(t){var e=t.attr("transform").split("translate("),n=w(e,2),r=(n[0],n[1].split(",")),o=w(r,2),a=o[0],i=o[1];return i.split(")"),[parseFloat(a),parseFloat(i)]}function E(t){var e,n,r,o,i,l,u,s,f,h,g="d3sm-lasso",p=!1,m=[],v=[],y=d3.line().x(function(t,e){return void 0!=s?s(t[0]):t[0]}).y(function(t,e){return void 0!=f?f(t[1]):t[1]}).curve(d3.curveLinearClosed),x=0,b=10,k="#17a2b8",w="10s",S=.3,j="5, 10",z="black",A=2,E="white",O="black",L=3,C=1e3,F=d3.easeExp;function Q(){var t,e;p&&((t=(t=c(n,"g","lasso-container").selectAll('path[instance="'+x+'"]')).data(v)).exit().remove(),e=t.enter().append("path"),V(t=t.merge(e).transition().duration(C).ease(F)))}function B(){var t=c(n,"g","lasso-container");t.selectAll('path[instance="'+x+'"]').remove();t.remove(),n.selectAll(r).classed("in-lasso",!1),Y()}function q(t){t.preventDefault(),t.stopPropagation();var r=c(n,"g","lasso-container");m=[],e.node().addEventListener("mousemove",D),e.node().addEventListener("mouseup",function(t){e.node().removeEventListener("mousemove",D),v.push(m),v=d(v)}),V(h=r.append("path").data([m]))}function V(t){t.attr("class",a(g,"lasso-path")).style("opacity",S).attr("fill",k).attr("d",y).attr("instance",x).style("stroke-dasharray",j).attr("stroke",z).attr("stroke-width",A).style("animation","lassoDash "+w+" linear").style("animation-iteration-count","infinite")}function D(t){if(void 0!=u&&u.dispatch(a(g,"drag")),1==t.which){d3.event=t;var r=d3.mouse(n.node());r=d3.mouse(e.node());if(void 0!=s&&(r[0]=s.invert(r[0])),void 0!=f&&(r[1]=f.invert(r[1])),r[0]=r[0]-i[0]-l[0],r[1]=r[1]-i[1]-l[1],m.length){var o=m[m.length-1],c=[r[0],r[1]],d=[o[0],o[1]];s&&(d[0]=s(d[0]),c[0]=s(c[0])),f&&(d[1]=f(d[1]),c[1]=f(c[1])),function(t,e){var n=t[0]-e[0],r=t[1]-e[1];return Math.sqrt(n*n+r*r)}(d,c)>b&&_(r)}else _(r)}}function _(t){if(void 0!=t){if(m.push(t),h.attr("d",y),m.length<3)return;K(v.concat([m]))}else K(v)}function K(t){return void 0==t&&(t=v),n.selectAll(r).each(function(e,n){var r=d3.select(this),o=r.absolutePosition(),a=[[o.left-i[0]-l[0],o.top-i[1]-l[1]],[o.right-i[0]-l[0],o.top-i[1]-l[1]],[o.left-i[0]-l[0],o.bottom-i[1]-l[1]],[o.right-i[0]-l[0],o.bottom-i[1]-l[1]]];void 0!=s&&(a[0][0]=s.invert(a[0][0]),a[1][0]=s.invert(a[1][0]),a[2][0]=s.invert(a[2][0]),a[3][0]=s.invert(a[3][0])),void 0!=f&&(a[0][1]=f.invert(a[0][1]),a[1][1]=f.invert(a[1][1]),a[2][1]=f.invert(a[2][1]),a[3][1]=f.invert(a[3][1]));var c=!1;for(n=0;nwindow.innerWidth&&o.style("left",d3.event.pageX-15-300+"px")}function ot(t,e){var n=d3.select(this).style("fill","black");if(d3.select(n.node().parentNode).select("line."+a(B,"tick")).attr("stroke",K).attr("stroke-width",Y),M){var o=d3.select(n.node().parentNode).select("line."+a(B,"guideline")),i=o.attr("minor");o.attr("stroke",function(t,e){return"true"==i?r(I,.8):I}).attr("stroke-width",function(t,e){return"true"==i?.8*J:J})}d3.select("#"+a(B,"guideline-tooltip")).remove()}return nt.label=function(t){return arguments.length?(m=t,nt):m},nt.tickTickLabelSpacer=function(t){return arguments.length?(X=t,nt):X},nt.tickLabelMargin=function(t){return arguments.length?(R=t,nt):R},nt.selection=function(e){return arguments.length?(t=e,nt):t},nt.orient=function(t){return arguments.length?(b=t,nt):b},nt.spaceX=function(t){return arguments.length?(w=t,nt):w},nt.spaceY=function(t){return arguments.length?(j=t,nt):j},nt.overflowQ=function(t){return arguments.length?(z=t,nt):z},nt.categoricalQ=function(t){return arguments.length?(A=t,nt):A},nt.guideLinesQ=function(t){return arguments.length?(M=t,nt):M},nt.grouping=function(t){return arguments.length?(e=t,nt):e},nt.scale=function(t){return arguments.length?(E=t,nt):E},nt.domainPadding=function(t){return arguments.length?(O=t,nt):O},nt.objectSpacer=function(t){return arguments.length?(L=t,nt):L},nt.minObjectSize=function(t){return arguments.length?(C=t,nt):C},nt.maxObjectSize=function(t){return arguments.length?(F=t,nt):F},nt.namespace=function(t){return arguments.length?(B=t,nt):B},nt.backgroundFill=function(t){return arguments.length?(Q=t,nt):Q},nt.objectClass=function(t){return arguments.length?(q=t,nt):q},nt.tickLabels=function(t){return arguments.length?(n=t,nt):n},nt.tickValues=function(t){return arguments.length?(o=t,nt):o},nt.numberOfTicks=function(t){return arguments.length?(V=t,nt):V},nt.lineStroke=function(t){return arguments.length?(D=t,nt):D},nt.lineStrokeWidth=function(t){return arguments.length?(_=t,nt):_},nt.tickStroke=function(t){return arguments.length?(K=t,nt):K},nt.tickStrokeWidth=function(t){return arguments.length?(Y=t,nt):Y},nt.tickLength=function(t){return arguments.length?(P=t,nt):P},nt.tickLabelFontSize=function(t){return arguments.length?(W=t,nt):W},nt.tickLabelMinFontSize=function(t){return arguments.length?(N=t,nt):N},nt.tickLabelMaxFontSize=function(t){return arguments.length?(T=t,nt):T},nt.tickLabelTextAnchor=function(t){return arguments.length?(l=t,nt):l},nt.tickLabelRotation=function(t){return arguments.length?(d=t,nt):d},nt.tickLabelFunc=function(t){return arguments.length?(G=t,nt):G},nt.tickLabelOnClick=function(t){return arguments.length?(H=t,nt):H},nt.guidelineSpace=function(t){return arguments.length?(h=t,nt):h},nt.guideLineStroke=function(t){return arguments.length?(I=t,nt):I},nt.guideLineStrokeWidth=function(t){return arguments.length?(J=t,nt):J},nt.transitionDuration=function(t){return arguments.length?(U=t,nt):U},nt.easeFunc=function(t){return arguments.length?($=t,nt):$},nt.objectSize=function(t){return arguments.length?(g=t,nt):g},nt.spacerSize=function(t){return arguments.length?(p=t,nt):p},nt.roundTo=function(t){return arguments.length?(tt=t,nt):tt},nt.reverseScaleQ=function(t){return arguments.length?(et=t,nt):et},nt.tickLabelOnHoverFunc=function(t){return arguments.length?(Z=t,nt):Z},nt},O.bar=function(t){var e,n,r,o,a,i,l,u,s="horizontal",d=!1,h=function(t,n){return e[t]},g=function(t,n){return d3.descending(e[t],e[n])},p=d3.scaleLinear(),m=.5,b=.05,w=50,A=100,M=2,E=j(),O="transparent",L="d3sm-bar",C="bar",F=1e3,Q=d3.easeExp,B=z(),q=1;function V(){var j="horizontal"==s||"bottom"==s||"top"==s,z=!j,V=v(t,L,{x:0,y:0,width:n,height:r},O);a=d3.keys(e),i=a.map(h);var D=void 0==o?a.sort(g):o,_=(a=f(D)).length,K=[Math.min.apply(Math,S(i))-m,Math.max.apply(Math,S(i))+m];p.domain(K).range(j?[0,r]:"right"==s?[0,n]:[n,0]);var Y=j?n:r;l=void 0==l?y(Y,_,w,A,b,d):l,u=void 0==u?x(a,Y,l,_,b,d):u;var P=k().horizontalQ(j).scale(p).moveby("category").numberOfObjects(_).objectClass(C).objectSize(l).spacerSize(u).transitionDuration(F).easeFunc(Q).namespace(L),X=P.exitFunction();P.exitFunction(function(t){void 0==l&&console.log(t.nodes(),l),X(t),t.selectAll("g").classed("to-remove",!0),t.selectAll("* > rect").transition().duration(F).attr("transform",function(t,e){return"translate(0,"+(z?0:p(K[1]))+")"}).attr("width",j?l:0).attr("height",z?l:0).remove()}),P(V,D,0);var R=[];V.selectAll("g:not(.to-remove)."+C).each(function(t,e){R.push(Number(d3.select(this).attr("parent-index")))}),E="index"==E.colorBy()?E.dataExtent([0,Math.max.apply(Math,R)]):E.dataExtent(K),V.selectAll("g."+C+":not(.to-remove)").each(function(t,n){var r=d3.select(this),o=(e[t],h(t,n)),a=(n=void 0==r.attr("parent-index")?n:r.attr("parent-index"),E(t,o,n,"fill")),i=E(t,o,n,"stroke"),u=c(r,"rect","bar-rect");void 0==u.attr("transform")&&u.attr("transform",function(t,e){return"translate(0,"+(z?0:p(K[1]))+")"}).attr("width",j?l:0).attr("height",z?l:0),u.transition().duration(F).ease(Q).attr("transform",function(t,e){return"translate("+(j?l-l*q:"right"==s?p(K[1])-p(o):l-l*q)+","+(z?l-l*q:p(K[1])-p(o))+")"}).attr("width",j?l*q:p(o)).attr("height",z?l*q:p(o)).attr("fill",a).attr("stroke",i).attr("stroke-width",M),r.on("mouseover",function(t,e){V.selectAll("g."+C).style("opacity",.2),r.style("opacity",1),u.attr("stroke-width",2*M)}),r.on("mouseout",function(){V.selectAll("g."+C).style("opacity",1),u.attr("stroke-width",M)})}),B.selection(V.selectAll(".bar-rect")).data(e),B()}return V.selection=function(e){return arguments.length?(t=e,V):t},V.data=function(t){return arguments.length?(e=t,V):e},V.orient=function(t){return arguments.length?(s=t,V):s},V.spaceX=function(t){return arguments.length?(n=t,V):n},V.spaceY=function(t){return arguments.length?(r=t,V):r},V.overflowQ=function(t){return arguments.length?(d=t,V):d},V.grouping=function(t){return arguments.length?(o=t,V):o},V.valueExtractor=function(t){return arguments.length?(h=t,V):h},V.sortingFunction=function(t){return arguments.length?(g=t,V):g},V.scale=function(t){return arguments.length?(p=t,V):p},V.domainPadding=function(t){return arguments.length?(m=t,V):m},V.objectSpacer=function(t){return arguments.length?(b=t,V):b},V.minObjectSize=function(t){return arguments.length?(w=t,V):w},V.maxObjectSize=function(t){return arguments.length?(A=t,V):A},V.barStrokeWidth=function(t){return arguments.length?(M=t,V):M},V.colorFunction=function(t){return arguments.length?(E=t,V):E},V.backgroundFill=function(t){return arguments.length?(O=t,V):O},V.namespace=function(t){return arguments.length?(L=t,V):L},V.objectClass=function(t){return arguments.length?(C=t,V):C},V.transitionDuration=function(t){return arguments.length?(F=t,V):F},V.easeFunc=function(t){return arguments.length?(Q=t,V):Q},V.barKeys=function(t){return arguments.length?(a=t,V):a},V.barValues=function(t){return arguments.length?(i=t,V):i},V.objectSize=function(t){return arguments.length?(l=t,V):l},V.spacerSize=function(t){return arguments.length?(u=t,V):u},V.tooltip=function(t){return arguments.length?(B=t,V):B},V.barPercent=function(t){return arguments.length?(q=t,V):q},V},O.bubbleHeatmap=function(t){var e,n,r,o,i,l,u,s,f,h,g,p,b="x",w="y",A="r",M="v",E=function(t,n){return e[t][b]},O=function(t,n){return e[t][w]},L=function(t,n){return e[t][A]},C=function(t,n){return e[t][M]},F=!1,Q=d3.scaleLinear(),B=.5,q=0,V=50,D=100,_=2,K="transparent",Y="d3sm-bubble",P="bubble",X=1e3,R=d3.easeExp,W=function(t,e){return E(t)-E(e)},N=function(t,e){return O(t)-O(e)},T=j().colorBy("value"),G=z();function H(){var b=v(t,Y,{x:0,y:0,width:n,height:r},K);(o=d3.keys(e)).sort(function(t,e){return W(t,e)||N(t,e)}),m("bubbleHeatmap","cells are sorted by",o),i=d(o.map(E)),l=d(o.map(O)),u=d(o.map(L)),s=d(o.map(C)),m("bubbleHeatmap","x and y keys are",{x:i,y:l});var w=i.length,j=l.length,z=[Math.min.apply(Math,S(u))-B,Math.max.apply(Math,S(u))+B];g=y(r,j,V,D,q,F),f=y(n,w,V,D,q,F),p=x(l,r,g,j,q,F),h=x(i,n,f,w,q,F),m("bubbleHeatmap","size of",{x:f,y:g}),Q.domain(z).range([Math.min(V/2,Math.min(g,f)/2),Math.min(g,f)/2]);var A=k().horizontalQ(!1).moveby("category").numberOfObjects(j).objectClass(a(P,"row")).objectSize(g).spacerSize(p).transitionDuration(X).easeFunc(R).namespace("row"),M=k().horizontalQ(!0).moveby("category").numberOfObjects(w).objectClass(P).objectSize(f).spacerSize(h).transitionDuration(X).easeFunc(R);A(b,l,0),b.selectAll("g."+a(P,"row")).each(function(t,e){M(d3.select(this),i,0)});var H=b.selectAll("g:not(.to-remove)."+P).data(o),Z=[];H.each(function(t,e){Z.push(Number(d3.select(this).attr("parent-index")))}),T="index"==T.colorBy()?T.dataExtent([0,Math.max.apply(Math,Z)]):T.dataExtent([0,Math.max.apply(Math,S(s))]),H.each(function(t,n){m("bubbleHeatmap","each cell",{key:t,index:n,node:d3.select(this).node()});var r=d3.select(this),o=(e[t],C(t,n)),i=L(t,n),l=(n=void 0==r.attr("parent-index")?n:r.attr("parent-index"),T(t,o,n,"fill")),u=T(t,o,n,"stroke");m("bubbleHeatmap","radius",{radius:i,scaled:Q(i),extent:z,range:Q.range()}),c(r,"circle",a(P,"circle")).attr("cx",f/2).attr("cy",g/2).attr("r",Q(i)).attr("fill",l).attr("stroke",u).attr("stroke-width",_)}),G.selection(H.selectAll("circle."+a(P,"circle"))).data(e),G()}return H.selection=function(e){return arguments.length?(t=e,H):t},H.data=function(t){return arguments.length?(e=t,H):e},H.spaceX=function(t){return arguments.length?(n=t,H):n},H.spaceY=function(t){return arguments.length?(r=t,H):r},H.xKey=function(t){return arguments.length?(b=t,H):b},H.yKey=function(t){return arguments.length?(w=t,H):w},H.rKey=function(t){return arguments.length?(A=t,H):A},H.vKey=function(t){return arguments.length?(M=t,H):M},H.cellKeys=function(t){return arguments.length?(o=t,H):o},H.xValues=function(t){return arguments.length?(i=t,H):i},H.yValues=function(t){return arguments.length?(l=t,H):l},H.rValues=function(t){return arguments.length?(u=t,H):u},H.vValues=function(t){return arguments.length?(s=t,H):s},H.xExtractor=function(t){return arguments.length?(E=t,H):E},H.yExtractor=function(t){return arguments.length?(O=t,H):O},H.rExtractor=function(t){return arguments.length?(L=t,H):L},H.vExtractor=function(t){return arguments.length?(C=t,H):C},H.overflowQ=function(t){return arguments.length?(F=t,H):F},H.scale=function(t){return arguments.length?(Q=t,H):Q},H.domainPadding=function(t){return arguments.length?(B=t,H):B},H.objectSpacer=function(t){return arguments.length?q=t:e},H.minObjectSize=function(t){return arguments.length?(V=t,H):V},H.maxObjectSize=function(t){return arguments.length?(D=t,H):D},H.bubbleStrokeWidth=function(t){return arguments.length?(_=t,H):_},H.backgroundFill=function(t){return arguments.length?(K=t,H):K},H.namespace=function(t){return arguments.length?(Y=t,H):Y},H.objectClass=function(t){return arguments.length?(P=t,H):P},H.transitionDuration=function(t){return arguments.length?(X=t,H):X},H.easeFunc=function(t){return arguments.length?(R=t,H):R},H.tooltip=function(t){return arguments.length?(G=t,H):G},H.colorFunction=function(t){return arguments.length?(T=t,H):T},H.xSize=function(t){return arguments.length?(f=t,H):f},H.xSpacerSize=function(t){return arguments.length?(h=t,H):h},H.ySize=function(t){return arguments.length?(g=t,H):g},H.ySpacerSize=function(t){return arguments.length?(p=t,H):p},H},O.heatmap=function(t){var e,n,r,o,i,l,u,s,f,h,g,p="x",b="y",w="v",A=function(t,n){return e[t][p]},M=function(t,n){return e[t][b]},E=function(t,n){return e[t][w]},O=!1,L=0,C=50,F=50,Q=100,B=100,q=2,V="transparent",D="d3sm-heatmap",_="heatmap",K=1e3,Y=d3.easeExp,P=function(t,e){return i.indexOf(A(t))-i.indexOf(A(e))},X=function(t,e){return l.indexOf(M(t))-l.indexOf(M(e))},R=j().colorBy("category"),W=z();function N(){var p=v(t,D,{x:0,y:0,width:n,height:r},V);o=d3.keys(e),i=d(o.map(A)),l=d(o.map(M)),u=d(o.map(E)),o.sort(function(t,e){return P(t,e)||X(t,e)}),m("heatmap","cells are sorted by",o),m("heatmap","x and y keys are",{x:i,y:l});var b=i.length,w=l.length;h=y(r,w,C,B,L,O),s=y(n,b,F,Q,L,O),g=x(l,r,h,w,L,O),f=x(i,n,s,b,L,O),m("heatmap","size of",{x:s,y:h});var j=k().horizontalQ(!1).moveby("category").numberOfObjects(w).objectClass(a(_,"row")).objectSize(h+g).spacerSize(0).transitionDuration(K).easeFunc(Y).namespace("row"),z=k().horizontalQ(!0).moveby("category").numberOfObjects(b).objectClass(_).objectSize(s+f).spacerSize(0).transitionDuration(K).easeFunc(Y);j(p,l,0),p.selectAll("g."+a(_,"row")).each(function(t,e){z(d3.select(this),i,0)});var N=p.selectAll("g:not(.to-remove)."+_);if(o.length!=l.length*i.length){var T={};o.map(function(t,e){T[A(t)+"::"+M(t)]=t});for(var G=[],H=0;Hr?r:n}).attr("fill",h).attr("stroke",v).attr("stroke-width",Q).attr("transform",function(t,e){return"translate("+(g?l/2:A(u))+","+(j?l/2:A(R[1])-A(u))+")"}),k.transition().duration(_).ease(K).attr("d",function(t,e){var n=g?A(i)-A(a):l;return b(!1,0,0,j?A(i)-A(a):l,n,C,d)}).attr("transform",function(t,e){return"translate("+(g?0:A(i))+","+(j?0:A(R[1])-A(i))+")"}).attr("stroke","black").attr("stroke-width",B).attr("fill","none"),x.transition().duration(_).ease(K).attr("d",function(t,e){var n=g?A(f)-A(s):l;return b(!0,0,0,j?A(f)-A(s):l,n,C,d)}).attr("transform",function(t,e){return"translate("+(g?0:A(s))+","+(j?0:A(R[1])-A(f))+")"}).attr("stroke","black").attr("stroke-width",B).attr("fill","none")}),Y.selection(z.selectAll("g:not(.to-remove)."+D)).data(e),Y()}return P.selection=function(e){return arguments.length?(t=e,P):t},P.data=function(t){return arguments.length?(e=t,P):e},P.orient=function(t){return arguments.length?(d=t,P):d},P.spaceX=function(t){return arguments.length?(n=t,P):n},P.spaceY=function(t){return arguments.length?(r=t,P):r},P.overflowQ=function(t){return arguments.length?(h=t,P):h},P.grouping=function(t){return arguments.length?(o=t,P):o},P.quartilesKey=function(t){return arguments.length?(g=t,P):g},P.quartilesKeys=function(t){return arguments.length?(p=t,P):p},P.valueExtractor=function(t){return arguments.length?(m=t,P):m},P.sortingFunction=function(t){return arguments.length?(w=t,P):w},P.scale=function(t){return arguments.length?(A=t,P):A},P.domainPadding=function(t){return arguments.length?(M=t,P):M},P.objectSpacer=function(t){return arguments.length?(E=t,P):E},P.minObjectSize=function(t){return arguments.length?(O=t,P):O},P.maxObjectSize=function(t){return arguments.length?(L=t,P):L},P.whiskerWidthPercent=function(t){return arguments.length?(C=t,P):C},P.colorFunction=function(t){return arguments.length?(F=t,P):F},P.boxStrokeWidth=function(t){return arguments.length?(Q=t,P):Q},P.whiskerStrokeWidth=function(t){return arguments.length?(B=t,P):B},P.backgroundFill=function(t){return arguments.length?(q=t,P):q},P.namespace=function(t){return arguments.length?(V=t,P):V},P.objectClass=function(t){return arguments.length?(D=t,P):D},P.transitionDuration=function(t){return arguments.length?(_=t,P):_},P.easeFunc=function(t){return arguments.length?(K=t,P):K},P.boxKeys=function(t){return arguments.length?(a=t,P):a},P.boxValues=function(t){return arguments.length?(i=t,P):i},P.objectSize=function(t){return arguments.length?(l=t,P):l},P.spacerSize=function(t){return arguments.length?(u=t,P):u},P.tooltip=function(t){return arguments.length?(Y=t,P):Y},P},O.colorFunction=j,O.datatoggle=function(t){var e,n,r,o=function(){},i="d3sm-databar",l=!1,u=!1;d.xAxisSelectQ=function(t){return arguments.length?(l=t,d):l},d.yAxisSelectQ=function(t){return arguments.length?(u=t,d):u},d.xAxisOptions=function(t){return arguments.length?(e=t,d):e},d.yAxisOptions=function(t){return arguments.length?(n=t,d):n},d.data=function(t){return arguments.length?(r=t,d):r},d.updateFunction=function(t){return arguments.length?(o=t,d):o},d.namespace=function(t){return arguments.length?(i=t,d):i},d.currentKeys=function(){var t={};return d3.keys(s).map(function(e,n){t[e]=s[e].currentOption()}),t};var s={};function d(){var e=c(t,"div","d-inline-flex flex-row flex-wrap").selectAll("div."+a(i,"select-filter"));e.exit().remove();var n=(e=e.data(d3.keys(r))).enter().append("div").attr("class","select-filter");return(e=e.merge(n).style("margin-right","10px")).each(function(t,e){var n=A(d3.select(this)).data(r[t]).namespace(a(i,t)).selectionName(t);n(),s[t]=n}),t.selectAll("select").on("change",function(){o()}),d}return d},O.groupingSpacer=k,O.tooltip=z,O.scatter=function(t){var e,n,r,o,a,i,l,c=d3.scaleLinear(),u=.5,s=function(t,n){return e[t].x},d=d3.scaleLinear(),f=.5,h=function(t,n){return e[t].y},g=d3.scaleLinear(),p=.5,m=function(t,e){return 2},y=2,x=10,b=2,k=j(),w="transparent",A="d3sm-scatter",M="scatter-point",E=1e3,O=d3.easeExp,L=z();function C(){var j=v(t,A,{x:0,y:0,width:n,height:r},w);o=d3.keys(e),a=o.map(s),i=o.map(h),l=o.map(m),o.length;var z=[Math.min.apply(Math,S(a))-u,Math.max.apply(Math,S(a))+u],C=[Math.min.apply(Math,S(i))-f,Math.max.apply(Math,S(i))+f],F=[Math.min.apply(Math,S(l))-p,Math.max.apply(Math,S(l))+p];c.domain(z).range([0,n]),d.domain(C).range([r,0]),g.domain(F).range([y,x]);var Q=j.selectAll("."+M),B=(Q=Q.data(o)).enter().append("circle").attr("class",M).attr("cx",0).attr("cy",r).attr("r",0),q=Q.exit();(Q=Q.merge(B)).each(function(t,n){var r=d3.select(this),o=e[t],u=a[n],s=i[n],f=l[n],h=k(t,o,n,"fill"),p=k(t,o,n,"stroke");r.transition().duration(E).ease(O).attr("cx",c(u)).attr("cy",d(s)).attr("r",g(f)).attr("fill",h).attr("stroke",p).attr("stroke-width",b),r.on("mouseover",function(t,e){Q.style("opacity",.2),r.style("opacity",1),r.transition().duration(E/2).ease(O).attr("stroke-width",2*b).attr("r",1.5*g(f))}),r.node().addEventListener("mouseout",function(){j.selectAll("."+M).style("opacity",1),r.transition().duration(E/2).ease(O).attr("stroke-width",b).attr("r",g(f))})}),q.transition().duration(E).ease(O).attr("cx",0).attr("cy",r).attr("r",0).remove(),L.selection(Q).data(e),L()}return C.selection=function(e){return arguments.length?(t=e,C):t},C.data=function(t){return arguments.length?(e=t,C):e},C.spaceX=function(t){return arguments.length?(n=t,C):n},C.spaceY=function(t){return arguments.length?(r=t,C):r},C.scaleX=function(t){return arguments.length?(c=t,C):c},C.domainPaddingX=function(t){return arguments.length?(u=t,C):u},C.valueExtractorX=function(t){return arguments.length?(s=t,C):s},C.scaleY=function(t){return arguments.length?(d=t,C):d},C.domainPaddingY=function(t){return arguments.length?(f=t,C):f},C.valueExtractorY=function(t){return arguments.length?(h=t,C):h},C.scaleR=function(t){return arguments.length?(g=t,C):g},C.domainPaddingR=function(t){return arguments.length?(p=t,C):p},C.valueExtractorR=function(t){return arguments.length?(m=t,C):m},C.minRadius=function(t){return arguments.length?(y=t,C):y},C.maxRadius=function(t){return arguments.length?(x=t,C):x},C.pointStrokeWidth=function(t){return arguments.length?(b=t,C):b},C.colorFunction=function(t){return arguments.length?(k=t,C):k},C.backgroundFill=function(t){return arguments.length?(w=t,C):w},C.namespace=function(t){return arguments.length?(A=t,C):A},C.objectClass=function(t){return arguments.length?(M=t,C):M},C.transitionDuration=function(t){return arguments.length?(E=t,C):E},C.easeFunc=function(t){return arguments.length?(O=t,C):O},C.pointKeys=function(t){return arguments.length?(o=t,C):o},C.valuesX=function(t){return arguments.length?(a=t,C):a},C.valuesY=function(t){return arguments.length?(i=t,C):i},C.valuesR=function(t){return arguments.length?(l=t,C):l},C.tooltip=function(t){return arguments.length?(L=t,C):L},C},O.plotZoom=function(t,e,r){var o,i=20,l=void 0==t.orient?"horizontal":t.orient(),c=t.spaceX(),u=t.spaceY(),s=t.selection(),d=e.selection(),f=r.selection();function h(){var e=s.select("."+a(t.namespace(),"object-container")),r=n(e.attr("transform")),o=e.attr("transform","translate(0,0)");c=s.node().getBBox().width-.9*t.spaceX(),u=s.node().getBBox().height-.9*t.spaceY(),o.attr("transform","translate("+r[0]+","+r[1]+")"),m("plotZoom","setLocks",{xLock:c,yLock:u})}function g(){var g,p;h(),"2D"==l&&(g=!0,p=!0),"horizontal"==l&&(g=!0,p=!1),"vertical"==l&&(p=!0,g=!1);var m=d3.event.transform,v=s.node().getBBox(),y=d.node().getBBox(),x=d.node().getBBox();if(v.width,v.x,v.height,v.y,y.width,y.height,x.width,x.height,"wheel"==o){d3.event.preventDefault();var b=d3.event.deltaY*i,k=d3.event.shiftKey;(m="2D"==l?k?{k:1,x:b,y:0}:{k:1,x:0,y:b}:g?{k:1,x:b,y:0}:{k:1,x:0,y:b}).applyX=function(t){return t*this.k+-1*this.x},m.applyY=function(t){return t*this.k+-1*this.y}}var w=s.select("."+a(t.namespace(),"object-container")),S=d.select("."+a(e.namespace(),"object-container")),j=f.select("."+a(r.namespace(),"object-container")),z=n(w.attr("transform")),A=(n(S.attr("transform")),n(j.attr("transform")),g?m.applyX(z[0]):0);g&&(A=A<-c?(m.x=0,-c):(m.x=0,Math.min(A,0)));var M=p?m.applyY(z[1]):0;p&&(M=M<-u?(m.y=0,-u):(m.y=0,Math.min(M,0))),w.attr("transform","translate("+A+","+M+")"),g&&S.attr("transform","translate("+A+",0)"),p&&j.attr("transform","translate(0,"+M+")")}return d3.select(s.thisSVG()),g.eventType=function(t){return arguments.length?(o=t,g):o},g.wheelSpeed=function(t){return arguments.length?(i=t,g):i},g.orient=function(t){return arguments.length?(l=t,g):l},g.xLock=function(t){return arguments.length?(c=t,g):c},g.yLock=function(t){return arguments.length?(u=t,g):u},g.setLocks=h,g.reset=function(){var n=s.select("."+a(t.namespace(),"object-container")),o=d.select("."+a(e.namespace(),"object-container")),i=f.select("."+a(r.namespace(),"object-container"));n.attr("transform","translate(0,0)"),o.attr("transform","translate(0,0)"),i.attr("transform","translate(0,0)")},g},O.multiPlotZoom=function(t){var e,r=20,o=void 0==t.orient?"horizontal":t.orient(),i=t.spaceX(),l=t.spaceY(),c=t.selection(),u=(d3.select(c.thisSVG()),[]),s=[];function d(){var e=c.select("."+a(t.namespace(),"object-container")),r=n(e.attr("transform")),o=e.attr("transform","translate(0,0)");i=c.node().getBBox().width-t.spaceX(),l=c.node().getBBox().height-t.spaceY(),o.attr("transform","translate("+r[0]+","+r[1]+")"),m("plotZoom","setLocks",{xLock:i,yLock:l})}function f(){d();var f,h,g=u.map(function(t,e){return t.selection()}),p=s.map(function(t,e){return t.selection()});"2D"==o&&(f=!0,h=!0),"horizontal"==o&&(f=!0,h=!1),"vertical"==o&&(h=!0,f=!1);var m=d3.event.transform,v=c.node().getBBox();if(g.map(function(t,e){return t.node().getBBox()}),g.map(function(t,e){return t.node().getBBox()}),v.width,v.x,v.height,v.y,"wheel"==e){d3.event.preventDefault();var y=d3.event.deltaY*r,x=d3.event.shiftKey;(m="2D"==o?x?{k:1,x:y,y:0}:{k:1,x:0,y:y}:f?{k:1,x:y,y:0}:{k:1,x:0,y:y}).applyX=function(t){return t*this.k+-1*this.x},m.applyY=function(t){return t*this.k+-1*this.y}}var b=c.select("."+a(t.namespace(),"object-container")),k=g.map(function(t,e){return t.select("."+a(u[e].namespace(),"object-container"))}),w=p.map(function(t,e){return t.select("."+a(s[e].namespace(),"object-container"))}),S=n(b.attr("transform")),j=(k.map(function(t,e){return n(t.attr("transform"))}),w.map(function(t,e){return n(t.attr("transform"))}),f?m.applyX(S[0]):0);f&&(j=j<-i?(m.x=0,-i):(m.x=0,Math.min(j,0)));var z=h?m.applyY(S[1]):0;h&&(z=z<-l?(m.y=0,-l):(m.y=0,Math.min(z,0))),b.attr("transform","translate("+j+","+z+")"),f&&k.map(function(t,e){t.attr("transform","translate("+j+",0)")}),h&&w.map(function(t,e){t.attr("transform","translate(0,"+z+")")})}return f.eventType=function(t){return arguments.length?(e=t,f):e},f.wheelSpeed=function(t){return arguments.length?(r=t,f):r},f.orient=function(t){return arguments.length?(o=t,f):o},f.xLock=function(t){return arguments.length?(i=t,f):i},f.yLock=function(t){return arguments.length?(l=t,f):l},f.xComponents=function(t){return arguments.length?(u=t,f):u},f.yComponents=function(t){return arguments.length?(s=t,f):s},f.setLocks=d,f.reset=function(){var e=c.select("."+a(t.namespace(),"object-container")),n=xAxisSel.select("."+a(xAxis.namespace(),"object-container")),r=yAxisSel.select("."+a(yAxis.namespace(),"object-container"));e.attr("transform","translate(0,0)"),n.attr("transform","translate(0,0)"),r.attr("transform","translate(0,0)")},f},O.violin=function(t){var e,n,a,i,l,u,d,g,p="horizontal",m=!0,b=!0,w=function(t,n){return e[t]},A=function(t,n){return d3.descending(e[t],e[n])},M=d3.scaleLinear(),E=.5,O=.05,L=50,C=100,F=2,Q=j(),B=function(t,e,n,o,a){var i=d3.scaleLinear().domain([o,a]).range([-.25,.05]),l=r(n.replace("#",""),i(t)),c="stroke"==e?0:.25;return r(l.replace("#",""),c)},q=3,V=2,D="transparent",_="d3sm-violin",K="violin",Y=1e3,P=d3.easeExp,X=["Q0","Q1","Q2","Q3","Q4"],R=z().keys([X[4],X[3],X[2],X[1],X[0]]),W=z(),N=function(t,e){return e.points},T=function(t,e){return e[t].value};function G(){var r,u,w,j="horizontal"==p,G=v(t,_,{x:0,y:0,width:n,height:a},D),H=void 0==i?d3.keys(e).sort(A):i;l=f(H);var Z=function(){var t,e,n=!0,r=["Q0","Q1","Q2","Q3","Q4"];function a(a,i){var l=i[a],c=t(a,l),u=d3.keys(c),s=u.map(function(t,n){return e(t,c)}),d=o(s,r),f=d3.histogram()(s),h=f.map(function(t){return t.length}),g=n?{x:0,y:d3.min(s)}:{x:d3.min(s),y:0},p=n?{x:0,y:d3.max(s)}:{x:d3.max(s),y:0},m=f.map(function(t,e){return n?{y:t.length?d3.median(t):d3.median([t.x0,t.x1]),x:h[e]}:{x:t.length?d3.median(t):d3.median([t.x0,t.x1]),y:h[e]}});m=[g].concat(m).concat([p]),l.binned=f,l.frequencies=h,l.contour=m,l.quartiles=d,l.pointKeys=u,l.pointValues=s}return a.horizontalQ=function(t){return arguments.length?(n=t,a):n},a.quartileKeys=function(t){return arguments.length?(r=t,a):r},a.violinPointsExtractor=function(e){return arguments.length?(t=e,a):t},a.violinPointValueExtractor=function(t){return arguments.length?(e=t,a):e},a}().horizontalQ(j).quartileKeys(X).violinPointsExtractor(N).violinPointValueExtractor(T);l.map(function(t,n){Z(t,e)});var I=l.length,J=(r=[]).concat.apply(r,S(l.map(function(t,n){return e[t].quartiles[X[0]]}))),U=(u=[]).concat.apply(u,S(l.map(function(t,n){return e[t].quartiles[X[X.length-1]]}))),$=[Math.min.apply(Math,S(J))-E,Math.max.apply(Math,S(U))+E];M.domain($).range(j?[0,a]:[0,n]);var tt=j?n:a;d=y(tt,I,L,C,O,m),g=x(H,tt,d,I,O,m),k().horizontalQ(j).scale(M).moveby("category").numberOfObjects(I).objectClass(K).objectSize(d).spacerSize(g).transitionDuration(Y).easeFunc(P).namespace(_)(G,H,0);var et=[];G.selectAll("g:not(.to-remove)."+K).each(function(t,e){s(l,t)&&et.push(Number(d3.select(this).attr("parent-index")))}),Q="index"==Q.colorBy()?Q.dataExtent([0,Math.max.apply(Math,et)]):Q.dataExtent($);var nt=Math.max.apply(Math,S((w=[]).concat.apply(w,S(l.map(function(t,n){return d3.max(e[t].frequencies)}))))),rt=d3.scaleLinear().domain([0,nt]).range([0,d/2]),ot=d3.line().x(function(t,e){return j?-rt(t.x):M(t.x)}).y(function(t,e){return j?M($[1])-M(t.y):-rt(t.y)}).curve(d3.curveBasis),at=d3.line().x(function(t,e){return j?rt(t.x):M(t.x)}).y(function(t,e){return j?M($[1])-M(t.y):rt(t.y)}).curve(d3.curveBasis);G.selectAll("g:not(.to-remove)."+K).each(function(t,n){var r=d3.select(this),o=e[t];if(s(l,t)){n=void 0==r.attr("parent-index")?n:r.attr("parent-index");var a=Q(t,o,n,"fill"),i=Q(t,o,n,"stroke"),u=c(r,"g","area"),f=c(u,"path","left"),g=c(u,"path","right"),p=c(r,"g","quarts"),m=(c(p,"line","q3"),c(p,"line","q1"),o.quartiles[X[3]],o.quartiles[X[2]]);if(o.quartiles[X[1]],r.attr("transform",j?"translate("+d/2+",0)":"translate(0,"+d/2+")"),f.transition().duration(Y).attr("d",function(t,e){return ot(o.contour)}).attr("fill",a).attr("stroke",i).attr("stroke-width",F),g.transition().duration(Y).attr("d",function(t,e){return at(o.contour)}).attr("fill",a).attr("stroke",i).attr("stroke-width",F),u.node().addEventListener("mouseover",function(t,e){G.selectAll("g."+K).style("opacity",.2),r.style("opacity",1),f.attr("stroke-width",2*F),g.attr("stroke-width",2*F)}),u.node().addEventListener("mouseout",function(t,e){G.selectAll("g."+K).style("opacity",1),f.attr("stroke-width",F),g.attr("stroke-width",F)}),b){var v=c(r,"g","points"),y=v.selectAll(".point").data(o.pointKeys);y.on("mouseover",null),y.exit().transition().ease(P).duration(Y).attr("r",0).attr("cy",j?M($[1])-M(m):rt(0)).attr("cx",j?rt(0):M(m)).remove();var x=y.enter().append("circle").attr("class","point").attr("r",0).attr("cx",j?0:M(m)).attr("cy",j?M(m):0);y=y.merge(x),z().selection(y).data(N(t,o)).header(W.header()).keys(W.keys()).values(W.values())();var k=d3.min(o.pointValues),w=d3.max(o.pointValues);y.transition().duration(Y).ease(P).attr("r",q).attr("cy",function(t,e){var n=o.pointValues[e];if(j)return M($[1])-M(n);var r=h(o.binned,n),a=Math.random(),i=rt(a*o.frequencies[r]*.5);return Math.random()>.5?i:-i}).attr("cx",function(t,e){var n=o.pointValues[e];if(j){var r=h(o.binned,n),a=Math.random(),i=rt(a*o.frequencies[r]*.5);return Math.random()>.5?i:-i}return M(n)}).attr("stroke",function(t,e){return t=o.pointValues[e],B(t,"stroke",i,k,w)}).attr("fill",function(t,e){return t=o.pointValues[e],B(t,"fill",i,k,w)}).attr("stroke-width",V),v.selectAll("circle.point").on("mouseover",function(t,e){G.selectAll("g."+K).style("opacity",.2),r.style("opacity",1),f.attr("stroke-width",2*F),g.attr("stroke-width",2*F),G.selectAll(".point").style("opacity",.2),d3.select(this).style("opacity",1).attr("r",2*q).attr("stroke-width",2*V)}),v.selectAll("circle.point").on("mouseout",function(t,e){var n=document.createEvent("SVGEvents");n.initEvent("mouseout",!0,!0),u.node().dispatchEvent(n),G.selectAll(".point").style("opacity",1),d3.select(this).attr("stroke-width",V).attr("r",q)})}else cV.selectAll(".point").transition().duration(Y).ease(P).attr("r",0).attr("cy",j?M($[1])-M(m):rt(0)).attr("cx",j?rt(0):M(m)).remove()}}),R.selection(G.selectAll("g:not(.to-remove)."+K+" .area")),void 0==R.data()&&R.data(e),R(),void 0==R.values()&&R.values([function(t,e){return t.quartiles[e]},function(t,e){return t.quartiles[e]},function(t,e){return t.quartiles[e]},function(t,e){return t.quartiles[e]},function(t,e){return t.quartiles[e]}])}return G.violinPointsExtractor=function(t){return arguments.length?(N=t,G):N},G.violinPointValueExtractor=function(t){return arguments.length?(T=t,G):T},G.selection=function(e){return arguments.length?(t=e,G):t},G.data=function(t){return arguments.length?(e=t,G):e},G.orient=function(t){return arguments.length?(p=t,G):p},G.spaceX=function(t){return arguments.length?(n=t,G):n},G.spaceY=function(t){return arguments.length?(a=t,G):a},G.overflowQ=function(t){return arguments.length?(m=t,G):m},G.pointsQ=function(t){return arguments.length?(b=t,G):b},G.grouping=function(t){return arguments.length?(i=t,G):i},G.valueExtractor=function(t){return arguments.length?(w=t,G):w},G.sortingFunction=function(t){return arguments.length?(A=t,G):A},G.scale=function(t){return arguments.length?(M=t,G):M},G.domainPadding=function(t){return arguments.length?(E=t,G):E},G.objectSpacer=function(t){return arguments.length?(O=t,G):O},G.minObjectSize=function(t){return arguments.length?(L=t,G):L},G.maxObjectSize=function(t){return arguments.length?(C=t,G):C},G.objectStrokeWidth=function(t){return arguments.length?(F=t,G):F},G.colorFunction=function(t){return arguments.length?(Q=t,G):Q},G.pointColorFunc=function(t){return arguments.length?(B=t,G):B},G.pointRadius=function(t){return arguments.length?(q=t,G):q},G.pointStrokeWidth=function(t){return arguments.length?(V=t,G):V},G.backgroundFill=function(t){return arguments.length?(D=t,G):D},G.namespace=function(t){return arguments.length?(_=t,G):_},G.objectClass=function(t){return arguments.length?(K=t,G):K},G.transitionDuration=function(t){return arguments.length?(Y=t,G):Y},G.easeFunc=function(t){return arguments.length?(P=t,G):P},G.quartileKey=function(t){return arguments.length?(quartileKey=t,G):quartileKey},G.quartileKeys=function(t){return arguments.length?(X=t,G):X},G.violinKeys=function(t){return arguments.length?(l=t,G):l},G.violinValues=function(t){return arguments.length?(u=t,G):u},G.objectSize=function(t){return arguments.length?(d=t,G):d},G.spacerSize=function(t){return arguments.length?(g=t,G):g},G.tooltip=function(t){return arguments.length?(R=t,G):R},G.pointsTooltip=function(t){return arguments.length?(W=t,G):W},G},O.numericLegend=function(t){var e,n,r=0,o=1,l=j(),u="d3sm-linear-vertical-gradient",s=12,d="transparent",f="black",h=2;function g(){var g=v(t,u,{x:0,y:0,width:e,height:n},d),p=c(c(t,"defs"),"linearGradient").attr("x1","0%").attr("y1","100%").attr("x2","0%").attr("y2","0%").attr("id",a(u,"numerical-legend-gradient"));l.dataExtent([r,o]).colorBy("value").valueExtractor(function(t,e,n){return e}),p.selectAll("stop").data(l.colors()).enter().append("stop").attr("offset",function(t,e){return e/(l.colors().length-1)}).attr("stop-color",function(t){return t});var m=c(g,"rect","legend").attr("transform","translate(0,"+s+")").style("fill","url(#"+a(u,"numerical-legend-gradient")+")").attr("x",0).attr("y",0).attr("width",e).attr("height",n-2*s).on("mousemove",function(t,e){!function(t,e,n,d){var g=d3.scaleLinear().domain([0,n.attr("height")]).range([o,r]),p=d3.mouse(n.node()),m=i(g(p[1]),h),v=l(void 0,m,void 0,"stroke"),y=l(void 0,m,void 0,"fill");c(c(d3.select("body"),"div",a(u,"legend-tooltip")).attr("id",a(u,"legend-tooltip")).style("position","absolute").style("left",d3.event.pageX+15+"px").style("top",d3.event.pageY+15+"px").style("background-color",y).style("border-color",v).style("min-width",s*(String(o).split(".")[0].length+3)+"px").style("min-height",s*(String(o).split(".")[0].length+3)+"px").style("border-radius","50%").style("border-radius","5000px").style("display","flex").style("justify-content","center").style("text-align","middle").style("padding","2px").style("border-style","solid").style("border-width",2),"div").text(m).style("color",f).style("align-self","center")}(0,0,m,d3.select(this))}).on("mouseout",function(t,e){d3.select("#"+a(u,"legend-tooltip")).remove()});c(g,"text","min").text(i(r,2)).attr("text-anchor","middle").attr("font-size",s+"px").attr("transform",function(t,r){return"translate("+e/2+","+(n-s/4)+")"}),c(g,"text","max").text(i(o,2)).attr("text-anchor","middle").attr("font-size",s+"px").attr("transform",function(t,n){return"translate("+e/2+","+s+")"})}return g.min=function(t){return arguments.length?(r=t,g):r},g.max=function(t){return arguments.length?(o=t,g):o},g.spaceX=function(t){return arguments.length?(e=t,g):e},g.spaceY=function(t){return arguments.length?(n=t,g):n},g.namespace=function(t){return arguments.length?(u=t,g):u},g.fontSize=function(t){return arguments.length?(s=t,g):s},g.backgroundFill=function(t){return arguments.length?(d=t,g):d},g.colorFunction=function(t){return arguments.length?(l=t,g):l},g.textColor=function(t){return arguments.length?(f=t,g):f},g.roundTo=function(t){return arguments.length?(h=t,g):h},g},O.categoricLegend=function(t){var e,n,r,o,a,i="horizontal",l=!1,u=function(t,e){return n[t]},s=function(t,e){return d3.ascending(t,e)},d=.05,h=10,g=100,p=2,m=j(),b="transparent",w="d3sm-legend",S="legend",z=1e3,A=d3.easeExp;function M(){var n="horizontal"==i,u=!n,j=v(t,w,{x:0,y:0,width:r,height:o},b);m.dataExtent([0,e.length-1]).colorBy("categories").categoryExtractor(function(t,e,n){return e});var M=Math.min(r,o),E=e.length,O=void 0==a?e.sort(s):a,L=f(O),C=n?r:o,F=y(C,E,h,g,d,l),Q=x(L,C,F,E,d,l);k().horizontalQ(n).scale(scale).moveby("category").numberOfObjects(E).objectClass(S).objectSize(F).spacerSize(Q).transitionDuration(z).easeFunc(A).namespace(w)(j,O,0),M=Math.min(F,r,o)/2-p,j.selectAll("g:not(.to-remove)."+S).each(function(t,e){var o=d3.select(this),a=c(o,"circle"),i=m(void 0,t,e,"fill"),l=m(void 0,t,e,"stroke"),s=n?M+p:(r-2*M)/2+M,d=u?M+p:(r-2*M)/2+M;a.attr("r",M).attr("cx",s).attr("cy",d).attr("fill",i).attr("stroke",l).attr("stroke-width",p);var f=c(o,"text");f.text(t).attr("text-anchor","middle").attr("transform",function(t,e){return"translate("+s+","+(d+f.node().getBoundingClientRect().height/4)+")"})})}return M.categories=function(t){return arguments.length?(e=t,M):e},M.selection=function(e){return arguments.length?(t=e,M):t},M.data=function(t){return arguments.length?(n=t,M):n},M.orient=function(t){return arguments.length?(i=t,M):i},M.spaceX=function(t){return arguments.length?(r=t,M):r},M.spaceY=function(t){return arguments.length?(o=t,M):o},M.overflowQ=function(t){return arguments.length?(l=t,M):l},M.grouping=function(t){return arguments.length?(a=t,M):a},M.valueExtractor=function(t){return arguments.length?(u=t,M):u},M.sortingFunction=function(t){return arguments.length?(s=t,M):s},M.objectSpacer=function(t){return arguments.length?(d=t,M):d},M.minObjectSize=function(t){return arguments.length?(h=t,M):h},M.maxObjectSize=function(t){return arguments.length?(g=t,M):g},M.bubbleStrokeWidth=function(t){return arguments.length?(p=t,M):p},M.colorFunction=function(t){return arguments.length?(m=t,M):m},M.backgroundFill=function(t){return arguments.length?(b=t,M):b},M.namespace=function(t){return arguments.length?(w=t,M):w},M.objectClass=function(t){return arguments.length?(S=t,M):S},M.transitionDuration=function(t){return arguments.length?(z=t,M):z},M.easeFunc=function(t){return arguments.length?(A=t,M):A},M},O.lasso=E,O.lassoWidget=function(t){var e,n,r,o,i,l,u,s,d="d3sm-lasso",f=(t=t,d3.line().x(function(t,e){return t[0]}).y(function(t,e){return t[1]}).curve(d3.curveLinearClosed),j());function h(t,e){console.log(t,e)}function g(){d3.select("html").select("style."+d+"lasso-widget").empty()&&d3.select("html").append("style").classed(d+"lasso-widget",!0).html("."+a(d,"data-table")+"{ height:100px; overflow:auto; }")}function p(t){console.log(t)}function m(){p((t.select("."+a(d,"tab-list")),t.select("."+a(d,"tab-content")).selectAll("."+a(d,"data-table")),{}))}function v(e,n){var r,o;t.select("#"+a(d,"tab",e)),t.select("#"+a(d,"tab","pane",e)),function(){var e=t.select("."+a(d,"tab-list")),n=(t.select("."+a(d,"tab-content")),e.selectAll("."+a(d,"tab")));if(0==n.size());else{var r=n.nodes()[n.size()-1];d3.select(r).select("a").attr("href")}}(),r=t.select("."+a(d,"tab-list")),o=t.select("."+a(d,"tab-content")),r=r.selectAll("."+a(d,"tab")),o=o.selectAll("."+a(d,"tab","pane")),r.each(function(t,e){d3.select(this).datum(e).attr("id",a(d,"tab",e)).select("a").attr("href","#"+a(d,"tab","pane",e)).text(function(t,n){var r=d3.select(this).text();return"Group"==r.split(" ")[0]?"Group "+e:r})}),o.each(function(t,e){d3.select(this).datum(e).attr("id",a(d,"tab","pane",e)),c(d3.select(this),"p","lead").text(function(t,n){var r=d3.select(this).text();return"Group"==r.split(" ")[0]?"Group "+e:r}),c(d3.select(this),"button","remove-btn").on("click",v)})}function y(t,i){var h=c(t,"p","lead").text("Group "+i),g=c(c(t,"div","table-responsive").attr("class",a(d,"data-table")),"table","table").classed("table-sm",!0).classed("table-hover",!0),p=(c(g,"caption","caption").html("List of selected"),c(t,"div","text-right")),m=i%f.colors().length;i%2!=0&&(m=f.colors().length-1-m);var y,k,w,S=function(t){var e=c(t,"button","lasso-btn").classed("btn btn-info",!0).classed(d,!0);return c(e,"i","fa").classed("fa-hand-pointer-o",!0),c(e,"span").text("Lasso select"),e}(p),j=(y=i,k=f.colors()[m],E().namespace(d).svg(e).objectClass(o).chartContainer(n).objectContainer(r).instance(y).color(k).yScale(s).xScale(u));!function(t,e,r,o){r.eventCatcher(t),t.node().addEventListener("click",x),t.datum([r]).on(a(d,"render"),function(){d3.select(this).datum()[0].draw()}).on(a(d,"drag"),function(t){var e=n.selectAll(".in-lasso-"+t[0].instance()).nodes(),a=e.map(function(t,e){var n=l(d3.select(t).datum());return n.__node=t,n});b(o,a,r)}),e.on("click",function(){r.allPoints([]),r.currentPoints([]),t.dispatch(a(d,"drag"))})}(S,function(t){var e=c(t,"button","clear-btn").classed("btn btn-light",!0).classed(d,!0);return c(e,"i","fa").classed("fa-eraser",!0),c(e,"span").text("Clear selection"),e}(p),j,g),c(w=c(p,"button","remove-btn").classed("btn btn-danger",!0).on("click",v),"i","fa").classed("fa-trash-o",!0),c(w,"span").text("Remove group"),h.on("mouseover",function(){j.draw()}),h.on("mouseout",function(){j.remove()})}function x(){var e=d3.select(this),n=e.datum()[0];n.toggle();var r=n.activeQ();function o(t){t.target!=e.node()&&t.target!=e.select("span").node()&&t.target!=e.select("i").node()&&(n.toggle(!1),e.classed("btn-info",!0),e.classed("btn-warning",!1),e.select("span").text("Lasso select"))}n.activeQ()?(e.classed("btn-info",!r),e.classed("btn-warning",r),e.select("span").text("Lasso select (active)"),t.selectAll("."+d+".lasso-btn").dispatch(a(d,"render")),d3.select("html").node().addEventListener("mousedown",o)):(e.classed("btn-info",!r),e.classed("btn-warning",r),e.select("span").text("Lasso select"),d3.select("html").node().removeEventListener("mousedown",o))}function b(t,e,n){var r=c(c(t,"thead"),"tr"),o=c(t,"tbody"),a=function(t,e){var n=d3.keys(e[0]).filter(function(t){return"__node"!=t});n.length>0&&n.push("remove");var r=t.selectAll("th");return(r=r.data(n)).exit().remove(),r=r.merge(r.enter().append("th").attr("scope","col")).text(function(t,e){return t}),n}(r,e);(function(t,e){var n=t.selectAll("tr");return(n=n.data(e)).exit().remove(),n=n.merge(n.enter().append("tr"))})(o,e).each(function(r,o){var i=d3.select(this);!function(t,e,n,r,o,a){(t=t.data(e)).exit().remove(),(t=t.merge(t.enter().append("td"))).html(function(t,e){return"remove"!=t?n[t]:""}).classed("text-left",function(t,e){return"remove"==t}),t.select("i.fa-close").on("click",function(t,e){var i=n.__node,l=d3.select(i);l.classed("in-lasso",!1),l.classed("in-lasso-"+o.instance(),!1),r.map(function(t,e){t.__node==i&&(r.splice(e,1),b(a,r,o))})})}(i.selectAll("td"),a,r,e,n,t),i.on("mouseover",function(t,e){n.applyObjectAttributes(d3.select(t.__node),!0)}).on("mouseout",function(t,e){n.applyObjectAttributes(d3.select(t.__node),!1)})})}function k(e,n){var r=t.select("."+a(d,"tab-list")),o=t.select("."+a(d,"tab-content")),l=function(){for(var e=t.select("."+a(d,"tab-list")).selectAll("li").size()-3,n="Group "+e,r=[];r.includes(n);)n="Group "+(e+=1);return e}();if(r.selectAll("."+a(d,"tab")).size()!=i){!function(t,e){var n=t.insert("li",".right-aligned-tabs").classed("nav-item",!0).classed("alert",!0).classed("alert-secondary",!0).classed("alert-dismissable",!0).classed("fade",!0).classed("show",!0).attr("role","alert").attr("id",a(d,"tab",e)).classed(a(d,"tab"),!0);c(n,"a").attr("data-toggle","tab").text("Group "+e).attr("href","#"+a(d,"tab","pane",e)).on("dblclick",function(t,e){var n=d3.select(this);n.attr("contenteditable",!0),d3.select(n.node().parentNode).classed("alert-secondary",!1).classed("alert-warning",!0)}).on("blur",function(t,e){var n=d3.select(this);n.attr("contenteditable",!1),d3.select(n.node().parentNode).classed("alert-secondary",!0).classed("alert-warning",!1)}).on("input",function(t,e){var n=d3.select(this),r=n.text();d3.select(n.attr("href")).select("p.lead").text(r)})}(r,l);var u=function(t,e){return t.append("div").datum(e).attr("role","tabpanel").classed("tab-pane",!0).classed("text-left",!0).classed(a(d,"tab","pane"),!0).attr("id",a(d,"tab","pane",e))}(o,l);y(u,l),u.attr("id"),t.select("."+a(d,"tab-content"))}else h("only "+i+" allowed.","warning")}return w=c(t,"div","card"),S=c(c(w,"div","card-header"),"ul","nav").classed("nav-tabs card-header-tabs",!0).classed(a(d,"tab-list"),!0).attr("role","tablist"),z=c(c(w,"div","card-body"),"div","tab-content").classed(a(d,"tab-content"),!0),A=c(S,"div","right-aligned-tabs").classed("nav ml-auto ",!0),function(t){c(c(c(t,"li",a(d,"plus-tab")).classed("ml-auto",!0).classed("nav-item",!0).on("click",k),"a","nav-link"),"i","fa").classed("fa-plus fa-2x text-success",!0)}(A),function(t){c(c(c(t,"li",a(d,"send-tab")).classed("ml-auto",!0).classed("nav-item",!0).on("click",m),"a","nav-link"),"i","fa").classed("fa-paper-plane-o fa-2x text-primary",!0)}(A),function(t){c(c(c(t,"li",a(d,"close-tab")).classed("ml-auto",!0).classed("nav-item",!0).on("click",function(){d3.mouse(d3.select("html").node())}),"a","nav-link"),"i","fa").classed("fa-window-close-o fa-2x text-danger",!0)}(A),c(c(z,"div","tab-pane").classed(a(d,"default-tab"),!0).classed("active",!0).classed("text-left",!0),"div").html("Click to add a new group.
Click to submit for re-analysis.
Click to close the dataselect widget."),g.objectClass=function(t){return arguments.length?(o="."+t.replace(".",""),g):o},g.svg=function(t){return arguments.length?(e=t,g):e},g.submit=function(t){return arguments.length?(p=t,g):p},g.maxNumberOfGroups=function(t){return arguments.length?(i=t,g):i},g.onError=function(t){return arguments.length?(h=t,g):h},g.objectContainer=function(t){return arguments.length?(r=t,g):r},g.chartContainer=function(t){return arguments.length?(n=t,g):n},g.dataExtractor=function(t){return arguments.length?(l=t,g):l},g.xScale=function(t){return arguments.length?(u=t,g):u},g.yScale=function(t){return arguments.length?(s=t,g):s},g;var w,S,z,A},O.selectFilter=A,O.upset=function(t){var e,n,r,o,i,l,u,s,f,h,g,p="horizontal",m=!1,b=20,w=50,S=2,j="transparent",z="d3sm-upset",A="upset",M=1e3,E=d3.easeExp,O=function(t,n){return e[t].set},L=function(t,n){return e[t].intersection},C=function(t,n){return e[t].elements},F=.05,Q=.05,B=function(t,e){return i.indexOf(O(t))-i.indexOf(O(e))},q=function(t,e){return l.indexOf(L(t))-l.indexOf(L(e))};function V(){var S="horizontal"==p,C=!S,V=v(t,z,{x:0,y:0,width:n,height:r},j);o=d3.keys(e),i=d(o.map(O)).sort(),l=d(o.map(L)).sort().sort(function(t,e){return t.split(";").length-e.split(";").length}),S?o.sort(function(t,e){return q(t,e)||B(t,e)}):o.sort(function(t,e){return B(t,e)||q(t,e)});var D=S?l:i,_=S?i:l,K=S?D.length:_.length,Y=S?_.length:D.length;h=y(n,K,b,w,F,m),s=y(r,Y,b,w,Q,m),g=x(D,n,h,K,F,m),f=x(_,r,s,Y,Q,m);var P=k().horizontalQ(!1).moveby("category").numberOfObjects(Y).objectSize(s).spacerSize(f).transitionDuration(M).easeFunc(E),X=k().horizontalQ(!0).moveby("category").numberOfObjects(K).objectClass(A).objectSize(h).spacerSize(g).transitionDuration(M).easeFunc(E);C?(X.objectClass(A),P.namespace("across").objectClass(a(A,"across")),P(V,_,0),V.selectAll("g."+a(A,"across")).each(function(t,e){X(d3.select(this),D,0)})):(X.namespace("across").objectClass(a(A,"across")),P.objectClass(A),X(V,D,0),V.selectAll("g."+a(A,"across")).each(function(t,e){P(d3.select(this),_,0)}));var R=V.selectAll("g:not(.to-remove)."+A),W={};o.map(function(t,e){W[O(t)+"::"+L(t)]=t}),R.data(o),R.each(function(t,n){var r=d3.select(this);if(void 0!=t){e[t];var o=O(t,n),i=L(t,n);r.classed(i,!0),r.classed(o,!0),c(r,"circle",a(A,"circle")).attr("cx",h/2).attr("cy",s/2).attr("r",void 0==u?Math.min(h,s)/2:u).attr("fill",i.includes(o)?"black":"rgb(233,233,233)").attr("stroke","black").attr("in-intersection",i.includes(o))}})}return V.selection=function(e){return arguments.length?(t=e,V):t},V.data=function(t){return arguments.length?(e=t,V):e},V.orient=function(t){return arguments.length?(p=t,V):p},V.spaceX=function(t){return arguments.length?(n=t,V):n},V.spaceY=function(t){return arguments.length?(r=t,V):r},V.overflowQ=function(t){return arguments.length?(m=t,V):m},V.minObjectSize=function(t){return arguments.length?(b=t,V):b},V.maxObjectSize=function(t){return arguments.length?(w=t,V):w},V.circleStrokeWidth=function(t){return arguments.length?(S=t,V):S},V.backgroundFill=function(t){return arguments.length?(j=t,V):j},V.namespace=function(t){return arguments.length?(z=t,V):z},V.objectClass=function(t){return arguments.length?(A=t,V):A},V.transitionDuration=function(t){return arguments.length?(M=t,V):M},V.easeFunc=function(t){return arguments.length?(E=t,V):E},V.cellKeys=function(t){return arguments.length?(o=t,V):o},V.setValues=function(t){return arguments.length?(i=t,V):i},V.intersectionValues=function(t){return arguments.length?(l=t,V):l},V.xObjectSpacer=function(t){return arguments.length?(F=t,V):F},V.yObjectSpacer=function(t){return arguments.length?(Q=t,V):Q},V.radius=function(t){return arguments.length?(u=t,V):u},V.setExtractor=function(t){return arguments.length?(O=t,V):O},V.intersectionExtractor=function(t){return arguments.length?(L=t,V):L},V.elementExtractor=function(t){return arguments.length?(C=t,V):C},V.setKeySortingFunction=function(t){return arguments.length?(B=t,V):B},V.intersectionKeySortingFunction=function(t){return arguments.length?(q=t,V):q},V.yObjectSize=function(t){return arguments.length?(s=t,V):s},V.ySpacerSize=function(t){return arguments.length?(f=t,V):f},V.xObjectSize=function(t){return arguments.length?(h=t,V):h},V.xSpacerSize=function(t){return arguments.length?(g=t,V):g},V.intersectionTotals=function(){var t={};return l.map(function(e,n){t[e]={total:0}}),o.map(function(e,n){var r=C(e,n);0==t[L(e,n)].total&&(Array.isArray(r)?(t[L(e,n)].total+=r.length,t[L(e,n)].values=r):t[L(e,n)].total+=r)}),t},V.setTotals=function(){var t={};return i.map(function(e,n){t[e]={total:0}}),o.map(function(e,n){var r=C(e,n);Array.isArray(r)?t[O(e,n)].total+=r.length:t[O(e,n)].total+=r}),t},V},O.filterTable=function(t){var e,n="d3sm-filter-table",r=function(t,e,n){return n};function o(){var o=c(t,"div","filter-table").classed(a(n,"container"),!0),u=c(o,"div","filter-input-group").classed("input-group",!0),s=(c(c(c(u,"div","input-group-prepend"),"span","input-group-text").classed("search-button",!0),"i","fa fa-search"),c(u,"input","form-control").attr("placeholder","filter").attr("type","text")),d=c(c(u,"div","input-group-append"),"a","close-button").classed("btn btn-outline-secondary",!0),f=(c(d,"i","fa fa-close"),d),h=c(c(o,"div","table-responsive"),"table","table").classed("table-hover",!0).classed("table-bordered",!0).classed("table-striped",!0),g=c(c(h,"thead").classed("thead-dark",!0),"tr"),p=c(h,"tbody"),m=d3.keys(e),v=d3.keys(e[m[0]]);(function(t,e){var n=t.selectAll("th");(n=n.data(e)).exit().remove(),(n=n.merge(n.enter().append("th"))).attr("scope","col").text(function(t,e){return t})})(g,v),l(i(p,m),v),s.on("input",function(t,n){var o,a=s.property("value"),c=new RegExp(a,"gi");""==a?o=m:(o=[],m.map(function(t,n){var a=e[t],i=v.map(function(t,e){return r(a,t,a[t])}).join("").match(c);null==i||""==i.join("")||o.push(t)})),l(i(p,o),v)}),f.on("click",function(t,e){s.property("value","").dispatch("input")})}function i(t,e){var n=t.selectAll("tr");return(n=n.data(e)).exit().remove(),n=n.merge(n.enter().append("tr"))}function l(t,n){return t.each(function(t,o){var a=e[t],i=d3.select(this).selectAll("td");(i=i.data(n)).exit().remove(),(i=i.merge(i.enter().append("td"))).attr("scope",function(t,e){return 0==e}).html(function(t,e){return r(a,t,a[t])})}),t}return o.data=function(t){return arguments.length?(e=t,o):e},o.namespace=function(t){return arguments.length?(n=t,o):n},o.fieldFunction=function(t){return arguments.length?(r=t,o):r},o},O.uniqueElements=e,O.getTranslation=n,O.modifyHexidecimalColorLuminance=r,O.tickRange=u,O.quartiles=o,O.extractViolinValues=function(t,e,n,r,a,i){var l={};return t.map(function(t,c){var u=n(t,c,e),s=d3.histogram()(u),d=s.map(function(t){return t.length}),f=r?{y:d3.min(u),x:0}:{x:d3.min(u),y:0},h=r?{y:d3.max(u),x:0}:{x:d3.max(u),y:0},g=s.map(function(t,e){return r?{y:t.length?d3.median(t):d3.median([t.x0,t.x1]),x:d[e]}:{x:t.length?d3.median(t):d3.median([t.x0,t.x1]),y:d[e]}}),p=o(u,i),m={values:u,binned:s,frequencies:d,points:[f].concat(g).concat([h])};m[a]=p,l[t]=m}),l},O.hypenate=a,O.round=i,O.getContainingSVG=l,O.interpolateColors=function(){return d3.interpolateRgbBasis(arguments)},O.truncateText=function(t,e,n,r,o,a){var i=t.node().getBoundingClientRect();for(t.text(e);Math.max(i.width,i.height)>o-r&&(e=(e=String(e)).slice(0,e.length-1),t.text(e+"..."),i=t.node().getBoundingClientRect(),0!=e.length););},O.safeSelect=c,O.whichBin=h,O.unique=d,O.flatten=f,O.setupStandardChartContainers=function(t,e,n){var r=arguments.length>3&&void 0!==arguments[3]?arguments[3]:{top:.01,bottom:.01,left:.01,right:.01},o=arguments.length>4&&void 0!==arguments[4]?arguments[4]:{w:.8,h:.6},a=arguments.length>5&&void 0!==arguments[5]?arguments[5]:{y:.1,x:.1},i=arguments.length>6&&void 0!==arguments[6]?arguments[6]:{x:0,margin:0,pos:"left"};void 0==n&&(n={w:window.innerWidth,h:window.Height});var l={w:n.w*o.w,h:n.h*o.h},c={top:r.top*l.h,bottom:r.bottom*l.h,left:r.left*l.w,right:r.right*l.w},u=l.w-c.left-c.right,s=l.h-c.top-c.bottom,d={x:s*a.x,y:u*a.y},f={x:u-d.y-i.x-2*i.margin,y:s-d.x},h={x:i.margin+c.left+("left"==i.pos?0:f.x+d.y),y:c.top,w:i.x,h:f.y},g={x:d.y+c.left+("left"==i.pos?i.x+2*i.margin:0),y:c.top,w:d.y,h:f.y},p={x:d.y+c.left+("left"==i.pos?i.x+2*i.margin:0),y:c.top,w:f.x,h:f.y},m={x:d.y+c.left+("left"==i.pos?i.x+2*i.margin:0),y:c.top+f.y,w:f.x,h:d.x};return n=d3sm.safeSelect(t,"svg",e).style("width",l.w+"px").style("height",l.h+"px"),a=d3sm.safeSelect(n,"g",d3sm.hypenate(e,"axes")),i=d3sm.safeSelect(n,"g",d3sm.hypenate(e,"legend")).attr("transform","translate("+h.x+","+h.y+")"),{svg:{selection:n,rect:l},plot:{selection:d3sm.safeSelect(n,"g",d3sm.hypenate(e,"plot")).attr("transform","translate("+p.x+","+p.y+")"),rect:p},xAxis:{selection:d3sm.safeSelect(a,"g",d3sm.hypenate(e,"x-axis")).attr("transform","translate("+m.x+","+m.y+")"),rect:m},yAxis:{selection:d3sm.safeSelect(a,"g",d3sm.hypenate(e,"y-axis")).attr("transform","translate("+g.x+","+g.y+")"),rect:g},legend:{selection:i,rect:h}}},O.log=m,O.warn=function(t,e,n){!0===window.d3sm.debugQ&&console.warn("%c[d3sm::"+t+"]:\t"+e,["background: #ffd53e","border-radius: 5000px","padding: 0px 2px","font-size: 14px"].join(";")),console.table(n)},O.info=function(t,e,n){window.d3sm.debugQ&&console.info("%c[d3sm::"+t+"]:\t"+e,["background: #009ccd","border-radius: 5000px","padding: 0px 2px","font-size: 14px"].join(";")),console.table(n)},O.error=function(t,e,n){window.d3sm.debugQ&&console.error("[d3sm::"+t+"]:\t"+e+"\t%o",n)},O.consoleGroup=g,O.consoleGroupEnd=p,O.resizeDebounce=function(t,e){var n=function(t,e,n){var r;return function(){var o=this,a=arguments,i=n&&!r;clearTimeout(r),r=setTimeout(function(){r=null,n||t.apply(o,a)},e),i&&t.apply(o,a)}}(function(){t()},e);window.addEventListener("resize",n)},O.debugQ=!1,window.d3sm=O,t.default=O,t}({}); -//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"d3sm.min.v0.0.3.js","sources":["../../src/scripts/modules/helpers.js","../../src/scripts/modules/array-functions.js","../../src/scripts/modules/utils.js","../../src/scripts/modules/grouping-spacer.js","../../src/scripts/modules/color-function.js","../../src/scripts/modules/tooltip.js","../../src/scripts/modules/select-filter.js","../../src/scripts/modules/lasso.js","../../src/scripts/modules/d3-prototypes.js","../../src/scripts/main.js","../../src/scripts/modules/axis.js","../../src/scripts/modules/bar.js","../../src/scripts/modules/bubble-heatmap.js","../../src/scripts/modules/heatmap.js","../../src/scripts/modules/box-whisker.js","../../src/scripts/modules/data-toggle.js","../../src/scripts/modules/scatter.js","../../src/scripts/modules/plot-zoom.js","../../src/scripts/modules/multi-plot-zoom.js","../../src/scripts/modules/violin.js","../../src/scripts/modules/numeric-legend.js","../../src/scripts/modules/categorical-legend.js","../../src/scripts/modules/lasso-widget.js","../../src/scripts/modules/upset.js","../../src/scripts/modules/filter-table.js"],"sourcesContent":["// import {hasQ} from './array-functions';\n/*******************************************************************************\n**                                                                            **\n**                                                                            **\n**                                HELPERS                                     **\n**                                                                            **\n**                                                                            **\n*******************************************************************************/\n/**\n* Helper function for Array.filter to get unique elements of the array\n* @param {*} value current value as mapping over array (self)\n* @param {number} index current index in the array\n* @param {Array} self passed array from Array.filter method\n* @returns {boolean} whether or not value is the first of its kind (i.e. indexOf(value) == index)\n*/\nexport function uniqueElements(value, index, self) { return self.indexOf(value) === index; }\n\n/**\n* Extracts x and y of translate from transform property\n* @param {string} transform transform property of svg element\n* @returns {number[]} x, y of translate(x, y)\n*/\nexport function getTranslation(transform) {\n  // Create a dummy g for calculation purposes only. This will never\n  // be appended to the DOM and will be discarded once this function\n  // returns.\n  var g = document.createElementNS('http://www.w3.org/2000/svg', 'g');\n  // Set the transform attribute to the provided string value.\n  transform = transform == undefined ? 'translate(0,0)' : transform;\n  g.setAttributeNS(null, 'transform', transform);\n  // consolidate the SVGTransformList containing all transformations\n  // to a single SVGTransform of type SVG_TRANSFORM_MATRIX and get\n  // its SVGMatrix.\n  var matrix = g.transform.baseVal.consolidate().matrix;\n  // As per definition values e and f are the ones for the translation.\n  return [matrix.e, matrix.f];\n}\n\n\n/**\n* Modifies luminance of hexidecimal number\n* @param {string} hex should be hexidecimal value with or without the proceeding octotrope\n* @param {number} lum value to increase or decrease luminosity by\n* @returns {string} updated hexidecimal value without the proceeding octotrope\n*/\nexport function modifyHexidecimalColorLuminance(hex, lum) {\n  // validate hex string\n  var hex = String(hex).replace(/[^0-9a-f]/gi, '');\n\n  if (hex.length < 6) {\n    hex = hex[0]+hex[0]+hex[1]+hex[1]+hex[2]+hex[2];\n\t}\n\tlum = lum || 0;\n\n\t// convert to decimal and change luminosity\n\tvar rgb = '#', c, i;\n\tfor (i = 0; i < 3; i++) {\n\t\tc = parseInt(hex.substr(i*2,2), 16);\n\t\tc = Math.round(Math.min(Math.max(0, c + (c * lum)), 255)).toString(16);\n\t\trgb += ('00'+c).substr(c.length);\n\t}\n\n\treturn rgb;\n}\n\n\n/**\n* @deprecated @see{@link tickRange}\n* @param {number} min\n* @param {number} max\n* @param {number} parts\n* @returns {number[]} array of length parts evenly partitioned between min and max\n*/\nexport function partitionRangeInto(min, max, parts) {\n  var diff = max - min\n  return Array(parts).map(function (e, i) { return min + diff / parts * i })\n}\n\n\n/**\n* Calculated the quartiles of the passed data and stores them with qKeys\n* @param {number[]} data list of numerical values\n* @param {string[]} [qKeys=['q0', 'q1', 'q2', 'q3', 'q4']] how returned object with quartiles should be stored\n* @returns {Object} with keys qKeys giving only the numerical values for the quartiles\n*/\nexport function quartiles(data, qKeys) {\n  var\n  q2 = d3.median(data),\n  lower = data.filter(x => x < q2),\n  upper = data.filter(x => x > q2),\n\n  q1 = d3.median(lower),\n  q1 = q1 == undefined ? q2 : q1,\n\n  q0 = d3.min(lower),\n  q0 = q0 == undefined ? q1 : q0,\n\n  q3 = d3.median(upper),\n  q3 = q3 == undefined ? q2 : q3,\n\n  q4 = d3.max(upper),\n  q4 = q4 == undefined ? q3 : q4,\n\n  k0 = 'q0', k1 = 'q1', k2 = 'q2', k3 = 'q3', k4 = 'q4',\n  obj = {}\n  if (qKeys!=undefined && qKeys.length == 5) { k0 = qKeys[0]; k1 = qKeys[1]; k2 = qKeys[2]; k3 = qKeys[3]; k4 = qKeys[4]; }\n  obj[k0] = q0; obj[k1] = q1; obj[k2] = q2; obj[k3] = q3; obj[k4] = q4;\n\n  return obj\n}\n\n\n/**\n* Helper function to get all values needed in making violin plots\n* @param {string[]} violinKeys\n* @param {number[]} data\n* @param {Function} valueExtractorFunction how to get values from data[violinKeys[i]]\n* @param {boolean} horizontalQ whether or not violins will be rendered horizontally or vertically\n* @param {string} qKey how the object containing the quartiles should be labeled as\n* @param {string[]} qKeys how each quartile should be labeled as\n* @returns {Object} required for @see{@link violin} containing keys values, binnned, frequencies, points, and quartiles\n* @see{@link quartiles}\n*/\nexport function extractViolinValues(\n  violinKeys,\n  data,\n  valueExtractorFunction,\n  horizontalQ,\n  qKey,\n  qKeys\n){\n  var obj = {}\n  violinKeys.map(function(k, i){\n     var d = valueExtractorFunction(k, i, data),\n     binned = d3.histogram()(d),\n     frequencies = binned.map(x=>x.length),\n     minPoint = horizontalQ ? {y: d3.min(d), x: 0} : {x: d3.min(d), y: 0},\n     maxPoint = horizontalQ ? {y: d3.max(d), x: 0} : {x: d3.max(d), y: 0},\n     points = binned.map(function(bin, i) {\n       return horizontalQ\n       ? {y: (bin.length) ? d3.median(bin): d3.median([bin.x0, bin.x1]), x: frequencies[i]}\n       : {x: (bin.length) ? d3.median(bin): d3.median([bin.x0, bin.x1]), y: frequencies[i]}\n     }),\n     quarts = quartiles(d, qKeys),\n     o = {\n       values: d,\n       binned: binned,\n       frequencies: frequencies,\n       points: [minPoint].concat(points).concat([maxPoint])\n     }\n     o[qKey] = quarts;\n     obj[k] = o;\n   });\n   return obj;\n}\n\n/**\n* Hypenates all strings together\n* @param {string[]} arguments\n* @returns {string} \"arg1-arg2-...-argn\"\n*/\nexport function hypenate(){ return Array.prototype.slice.call(arguments).join('-') }\n\n\n/**\n* Rounds decimals of number to precision\n* @param {number} number\n* @param {number} precision\n* @returns {number} rounded to precision\n*/\nexport function round(number, precision) {\n  var shift = function (number, precision, reverseShift) {\n    if (reverseShift) {\n      precision = -precision;\n    }\n    var numArray = ('' + number).split('e');\n    return +(numArray[0] + 'e' + (numArray[1] ? (+numArray[1] + precision) : precision));\n  };\n  return shift(Math.round(shift(number, precision, false)), precision, true);\n}\n\n/**\n* recursively ascends element.parentElement to find a svg tag\n* @param {Element} element\n* @returns {Element | undefined}\n*/\nexport function getContainingSVG(element) {\n  var parent = element.parentElement\n  var tag = parent.tagName.toLowerCase()\n  if (tag === 'svg') { return parent; }\n  if (tag === 'html') { return undefined; }\n  return getContainingSVG(parent);\n}\n\n/**\n* Maps arguments in to d3.interpolateRgbBasis\n* @param arguments\n* @returns {Function}\n*/\nexport function interpolateColors(){return d3.interpolateRgbBasis(arguments)}\n\n\n/**\n* Trys to reduce text to fit in specified area, made for tick labels as called by\n* @see{@link axis}\n* @param {d3.selection} t container for specific axis tick\n* @param {string} text to be the label of the passed axis tick\n* @param {boolean} orient of the axis, true is horizontal, false is vertical\n* @param {number} tickLength is the length of the text\n* @param {number} space is the amount of availble space for the text and the tick to fit in\n* @param {boolean} overflowQ whether or not allowed to go over the alloted space\n* @returns {none}\n*/\nexport function truncateText(t, text, orient, tickLength, space, overflowQ) {\n  var rect = t.node().getBoundingClientRect()\n  t.text(text)\n  while (Math.max(rect.width, rect.height) > space - tickLength) {\n    text = String(text)\n    text = text.slice(0, text.length - 1)\n    t.text(text + '...')\n    rect = t.node().getBoundingClientRect()\n    if (text.length == 0) break\n  }\n}\n\nexport function truncateString(string, space, font) {\n  var chars = space / font;\n  if (chars < string.length) {\n    return string.slice(0, Math.round(chars-5)) + '...'\n  } else {\n    return string\n  }\n}\n\n\n/**\n* Trys to use d3.selection to get element, if it doesnt exist, makes one\n* @param {d3.selection} sel selection in which to try and find object\n* @param {string} tag tag of which to try and select\n* @param {string} [cls=''] class of tag to try and grab\n* @returns {d3.selection} of either append or selected tag.cls within sel\n*/\nexport function safeSelect(sel, tag, cls) {\n  var clsStr = cls == undefined ? '' : '.'+cls;\n  var sSel = sel.select(tag+clsStr).empty()\n  ? sel.append(tag)\n  : sel.select(tag+clsStr)\n  return sSel\n  .classed(clsStr.replace('.', ''), true)\n  .attr('transform', sSel.attr('transform') == undefined ? 'translate(0,0)' : sSel.attr('transform'))\n}\n\n/**\n* evenly partitions the range [min, max] into n parts\n* @param {number} min\n* @param {number} max\n* @param {number} n\n* @returns {number[]} array of length n evenly partitioned between min and max\n*/\nexport function tickRange(min, max, n) {\n  var a = [min]\n  var d = max-min\n  var s = d / (n-1)\n  for (var i = 0; i < n-2; i++) { a.push(min + s * (i+1)) }\n  a.push(max)\n  return a\n}\n\n\nexport function euclideanDistance(p1, p2){\n  var a =  p1[0] - p2[0], b =  p1[1] - p2[1]\n  return Math.sqrt(a*a + b*b)\n}\n","import {uniqueElements} from './helpers';\n/*******************************************************************************\n**                                                                            **\n**                                                                            **\n**                              PROTOTYPES                                    **\n**                                                                            **\n**                                                                            **\n*******************************************************************************/\n/**\n* This function tests to see if all elements of the passed array are true.\n* @param {Array} array of values\n* @param {Function} [func function(value){return value == true;}] is applied to each value of the array and should return a boolean.\n* @returns {boolean} if all values are true by function\n*/\nexport function all( array, func ) {\n  if (func == undefined) { return array.every( function(value) { return value === true; }); }\n  return array.every( function(value) { return func(value); } );\n}\n\n/**\n* Counts the number of occurances of each element in the given array.\n* @param {Array} array of elements\n* @returns {Object} of key: value pairs where key is an element in the array and value is the number of times it occurs.\n*/\nexport function tally( array ) {\n  var tallies = {};\n  array.map( function ( element ) {\n    if ( hasQ(Object.keys(tallies), element) ) { tallies[element] = 1; }\n    else { tallies[element] += 1; }\n  });\n  return tallies;\n}\n/**\n* Short-hand for array.includes(item);\n* @param {Array} array\n* @param {*} item to test if contained in  {array}\n* @returns {boolean}\n*/\nexport function hasQ( array, item ) { return array.includes(item); }\n\n/**\n* Returns first item in array\n* @param {Array} array of items\n* @returns {*} array[0]\n*/\nexport function first( array ) { return array[0]; }\n\n/**\n* Returns last item in array\n* @param {Array} array of items\n* @returns {*} array[array.length-1]\n*/\nexport function last( array ) { return array[array.length-1]; }\n\n/**\n* Calculates the total value of numbers in passed array\n* @param {number[]} array of numerical values\n* @returns {number} sum over elements in array\n*/\nexport function total( array ) { return array.reduce((a, b) => a + b, 0) };\n\n/**\n* Removes duplicates in array\n* @param {Array} array of items\n* @returns {Array} of items such that item_i != item_j for all i < j\n* @see{@link uniqueElements} for the filtering function\n*/\nexport function unique( array ) { return array.filter( uniqueElements ); }\n\n/**\n* Filters passed array for specified indicies\n* @param {Array} array of items\n* @param {number[]} positions of integers such that i < array.length\n* @returns {Array} of items such that for any item_i, positions.includes(i) === true\n*/\nexport function get( array, positions ) {\n  return array.filter( function( value, index ) { return hasQ(positions, index); } );\n}\n\n/**\n* Determines if all elements in passed array are arrays themselves.\n* @param {Array} array of items\n* @returns {boolean} true if Array.isArray(e) is true for all e in array\n* @see{@link all}\n*/\nexport function listOfListsQ( array ) {\n  return all( array.map( function( element, index ) { return Array.isArray(element) } ) )\n}\n\n/**\n* Built on top of @see{@link get}, mapping if positions is a list of lists (@see{@link listOfListsQ})\n* @param {Array} array of items\n* @param {number[] | []number[] } positions of integers or list of positions of integers\n* @returns {boolean} returns specified positions from array. If nested positions passed, returns requested items in same structure.\n*/\nexport function cut( array, positions ) {\n  if ( listOfListsQ(array) ) { return positions.map(function(pos, i) { return array.get(pos); }); }\n  return get( array, positions );\n}\n\n/**\n* Given an array of objects, constructs new objects where each value is a list\n* based on the corresonding key, which is extracted by the parameter by\n* @param {Objects[]} array of objects\n* @param {string} by key within all objects of passed array\n* @param {string[]} [groups] saves some computation if all known values extracted by mapping over the parameter by are passed\n* @returns {Object} of key value pairs, where keys are all values of the key by from an object in the passed array and the value are those corresponding objects.\n*/\nexport function groupBy (array, by, groups) {\n  if (groups == undefined) {\n    groups = unique(array.map(function(elements, index){ return element[by]; }));\n    groups.map(function(value, index){groupped[value] = []})\n  }\n\n  var groupped = {};\n  array.map(function(element, index){groupped[element[by]].push(element)});\n  return groupped\n}\n\n/**\n* Tests if two arrays are equivalent\n* @param {Array} array\n* @param {Array} other\n* @returns {boolean} if every element of array matches that of other\n*/\nexport function arrayEquals(array, other) {\n  if (!other)\n      return false;\n  // compare lengths - can save a lot of time\n  if (array.length != other.length)\n      return false;\n\n  for (var i = 0, l=array.length; i < l; i++) {\n      // Check if we have nested arrays\n      if (array[i] instanceof Array && other[i] instanceof Array) {\n          // recurse into the nested arrays\n          if (!arrayEquals(array[i],other[i]))\n              return false;\n      }\n      else if (array[i] != other[i]) {\n          // Warning - two different object instances will never be equal: {x:20} != {x:20}\n          return false;\n      }\n  }\n  return true;\n}\n\n\n\n/**\n* Recursively tallies the number of elements at each level of the passed, putatively nested array\n* @param {Array} array of items which may include nested arrays\n* @param {number} [level=0] current depth in the recursion\n* @param {Array} [levelData=[]] keeps track of items seen so far at each depth\n* @returns {Array} stating the number of elements (array inclusive) found at each level of the array\n*/\nexport function elementsAtLevels(array, level, levelData) {\n  level = level == undefined ? 0 : level + 1;\n  levelData = levelData == undefined ? [] : levelData;\n  if ( level >= levelData.length ) { levelData.push(array.length)} else {levelData[level] += array.length }\n  array.map(function(e, i) {if (Array.isArray(e)){ elementsAtLevels(e, level, levelData) }})\n  return levelData\n}\n\n\n/**\n* Recursively tallies the number of elements of the passed, putatively nested array\n* @param {Array} array of items which may include nested arrays\n* @param {number} [elements=0] current number of elements seen so far\n* @returns {number} number of elements (array inclusive) found in passed array\n*/\nexport function numberOfElements( array, elements ) {\n  elements = elements == undefined ? 0 : elements;\n  array.map(function(e, i) {\n    if ( Array.isArray(e) ) { elements = numberOfElements(e, elements) }\n    else { elements += 1 }\n  })\n  return elements\n}\n\n/**\n* Concats all nested arrays in passed array to form a single array\n* @param {Array} array of putatively nested arrays\n* @param {Array} [flat=[]] current flattened array\n* @returns {Array} with every element in the same level\n*/\nexport function flatten( array, flat ) {\n  flat = flat == undefined ? [] : flat;\n  array.map(function(e, i){\n    if (Array.isArray(e)) {flat = flat.concat(flatten(e))}\n    else {flat.push(e)}\n  })\n  return flat;\n}\n\n/**\n* Search of list of lists to find which - if any - passed value is in\n* @param {Array[]} bins list of lists of values\n* @param {*} value item to test if in any of the bins\n* @returns {number} indicating the index of the bin in which value was found\n*/\nexport function whichBin(bins, value) {\n  var i = -1\n  for (var j = 0; j < bins.length; j++) { if (hasQ(bins[j],value)) {return j} }\n  return i\n}\n","import {hypenate, safeSelect} from './helpers';\nimport {total} from './array-functions';\n\n\n/**\n * calls console.group if d3sm.debugQ == true\n * @param {string} name of the group\n * @returns {undefined}\n */\nexport function consoleGroup(name) {\n  if (window.d3sm.debugQ === true){\n    console.group(name)\n  }\n}\n\n/**\n * calls console.groupEnd if d3sm.debugQ == true\n * @returns {undefined}\n */\nexport function consoleGroupEnd() {\n  if (window.d3sm.debugQ === true){\n    console.groupEnd()\n  }\n}\n\n/**\n * Calls console.log if d3sm.debugQ == true\n * @param {string} func name of the function logging\n * @param {string} msg to log\n * @param {Object} data to be logged along side the message\n * @returns {undefined}\n */\nexport function log(func, msg, data) {\n  if (window.d3sm.debugQ === true){\n    console.log(\n      `%c[d3sm::${func}]:\\t${msg}`,\n      [\n        'background: #6cd1ef',\n        'border-radius: 5000px',\n        'padding: 0px 2px',\n        'font-size: 14px'\n      ].join(';')\n    )\n    console.table(data)\n    // console.trace()\n  }\n}\n\n/**\n * Calls console.warn if d3sm.debugQ == true\n * @param {string} func name of the function warning\n * @param {string} msg to display\n * @param {Object} data to be displayed along side the message\n * @returns {undefined}\n */\nexport function warn(func, msg, data) {\n  if (window.d3sm.debugQ === true)\n    console.warn(\n      `%c[d3sm::${func}]:\\t${msg}`,\n      [\n        'background: #ffd53e',\n        'border-radius: 5000px',\n        'padding: 0px 2px',\n        'font-size: 14px'\n      ].join(';')\n    )\n    console.table(data)\n}\n/**\n * Calls the console.info if d3sm.debugQ == true\n * @param {string} func name of the function providing info\n * @param {string} msg to display\n * @param {Object} data to be displayed along side the message\n * @returns {undefined}\n */\nexport function info(func, msg, data) {\n  if (window.d3sm.debugQ)\n    console.info(\n      `%c[d3sm::${func}]:\\t${msg}`,\n      [\n        'background: #009ccd',\n        'border-radius: 5000px',\n        'padding: 0px 2px',\n        'font-size: 14px'\n      ].join(';')\n    )\n    console.table(data)\n}\n\n\n/**\n * Calls console.error if d3sm.debugQ == true\n * @param {string} func name of the function which sends the error\n * @param {string} msg to display\n * @param {Object} data to be displayed along side the message\n * @returns {undefined}\n */\nexport function error(func, msg, data) {\n  if (window.d3sm.debugQ)\n    console.error(`[d3sm::${func}]:\\t${msg}\\t%o`,data)\n}\n\n\n\n\n\n/**\n* Function for setting up containers for most plots with the y axis container\n* positioned on the left and the x axis container positioned on the bottom\n* @param {d3.selection} selection selection of container in which the svg is or should be made\n* @param {string} namespace namespace of the chart\n* @param {Object} [space={w:window.innerWidth, h:window.innerHeight}] the width (w) and height (h) availble\n* @param {number} [space.w=window.innerWidth] the available width in which to render the chart\n* @param {number} [space.h=window.innerHeight] the available height in which to render the chart\n\n* @param {Object} [margins={top: 0.01, bottom: 0.01, left: 0.01, right: 0.01}] the margins for the chart\n* @param {number} [margins.top=0.01] the top margin of the chart\n* @param {number} [margins.bottom=0.01] the bottom margin of the chart\n* @param {number} [margins.left=0.01] the left margin of the chart\n* @param {number} [margins.right=0.01] the right margin of the chart\n\n\n* @param {Object} [percentages = {axes:{x:0.1,y:0.1},space:{w:0.8,h:0.6}}] percentages of the paramater space of which to make the x and y axes as well as the percent of the availble space in which to render the plot\n* @param {Object} [percentages.axes={x:0.1,y:0.1}] the percentages of the paramater space, of which the x and y axes will take up\n* @param {number} [percentages.axes.xAxisPercent=0.1] the percentages of the paramater space, of which the x axis will take up\n* @param {number} [percentages.axes.yAxisPercent=0.1] the percentages of the paramater space, of which the y axis will take up\n\n* @param {Object} [percentages.space={w:0.8,h:0.6}] the percentages of the paramater space, of which the SVG's width and height will be set\n* @param {number} [percentages.space.percentOfSpaceForWidth=0.1] the percentages of the paramater space, of which the SVG's width will be set\n* @param {number} [percentages.space.percentOfSpaceForHeight=0.1] the percentages of the paramater space, of which the SVG's height will be set\n\n* @returns {Object} returns the selection and \"boundingRects\" of the plot container, x-axis container and y-axis container\n* as\n*\n* {\n*\n*   plot: {selection: plotSelection, rect: plotRect},\n*\n*   xAxis:{selection:xAxisSelection, rect:xAxisRect},\n*\n*   yAxis: {selection:yAxisSelection, rect:yAxisRect}\n*\n* }\n*\n* where each rect has form:\n*\n* {x: #, y: #, h: #, w: #}\n*\n* depicting the starting x and y coordinate of the coresponding container (also their default transform values) as well their height (h) ans width (w)\n*/\n// export function setupStandardChartContainers( selection, namespace, space, margins, percentages) {\n// export function setupStandardChartContainers(\n//   selection,\n//   namespace,\n//   space={w:availableWidth=window.innerWidth, h:availableHeight=window.innerHeight},\n//   margins={top:0.01, bottom:0.01, left:0.01, right:0.01},\n//   percentages={axes: {x: xAxisPercent=0.1, y: yAxisPercent=0.1}, space: {w: percentOfSpaceForWidth, h: percentOfSpaceForHeight}}\n// ) {\n//   if (space == undefined) { space = {w: window.innerWidth, h: window.innerHeight} }\n//   if (margins == undefined) { margins = {top: 0.01, bottom: 0.01, left: 0.01, right: 0.01} }\n//   if (percentages == undefined) { percentages = {}; }\n//   if (percentages.axes == undefined) { percentages.axes = { x:0.1, y:0.1 } }\n//   if (percentages.space == undefined) { percentages.space = { w: 0.8, h: 0.6 } }\n//\n//   // SVG width and height\n//   var svgSpace =  {\n//     w: space.w * percentages.space.w,\n//     h: space.h * percentages.space.h\n//   },\n//\n//   // Space after removing margins\n//   chartSpace = {\n//     w: svgSpace.w - (margins.left * space.w) - (margins.right * space.w),\n//     h: svgSpace.h - (margins.top * space.h) - (margins.bottom * space.h)\n//   },\n//\n//   // main dimension of x and y axies\n//   // e.g. defines how tall x axis is as length is determined by plotRect.w\n//   axesSpace = {\n//     x: chartSpace.h * percentages.axes.x,\n//     y: chartSpace.w * percentages.axes.y\n//   },\n//\n//   // space left for drawing the chart properly (e.g. bars, violins, etc)\n//   drawingSpace = {\n//     x: chartSpace.w - axesSpace.y,\n//     y: chartSpace.h - axesSpace.x\n//   },\n//\n//\n//   yAxisRect = {\n//     x: axesSpace.y + (margins.left * space.w),\n//     y: (margins.top * space.h),\n//     w: axesSpace.y,\n//     h: drawingSpace.y\n//   },\n//\n//   plotRect = {\n//     x: axesSpace.y + (margins.left * space.w),\n//     y: (margins.top * space.h),\n//     w: drawingSpace.x,\n//     h: drawingSpace.y\n//   },\n//\n//   xAxisRect = {\n//     x: axesSpace.y + (margins.left * space.w),\n//     y: (margins.top * space.h + plotRect.h),\n//     w: drawingSpace.x,\n//     h: axesSpace.x\n//   }\n//\n//\n//   var container = safeSelect(selection, 'svg', namespace)\n//     .style('width', svgSpace.w+'px')\n//     .style('height', svgSpace.h+'px')\n//\n//   var axes = safeSelect(container, 'g', hypenate(namespace, 'axes'))\n//\n//   // .attr('transform', \"translate(\"+plotRect.x+\",\"+plotRect.y+\")\"),\n//\n//   var plot = safeSelect(container, 'g', hypenate(namespace, 'plot'))\n//     .attr('transform', \"translate(\"+plotRect.x+\",\"+plotRect.y+\")\")\n//\n//   var xAxis = safeSelect(axes, 'g', hypenate(namespace, 'x-axis'))\n//     .attr('transform', \"translate(\"+xAxisRect.x+\",\"+xAxisRect.y+\")\")\n//\n//   var yAxis = safeSelect(axes, 'g', hypenate(namespace, 'y-axis'))\n//     .attr('transform', \"translate(\"+yAxisRect.x+\",\"+yAxisRect.y+\")\")\n//\n//   return {\n//     svg: {\n//       selection: container,\n//       rect: svgSpace\n//     },\n//     plot: {\n//       selection: plot,\n//       rect: plotRect\n//     },\n//     xAxis: {\n//       selection: xAxis,\n//       rect: xAxisRect\n//     },\n//     yAxis: {\n//       selection: yAxis,\n//       rect: yAxisRect\n//     }\n//   }\n//\n//   // return [plot, xAxis, yAxis]\n// }\n//\n//\n\n\n\n\n\nexport function setupStandardChartContainers(\n  selection,\n  namespace,\n  container,\n  margins={top:0.01, bottom:0.01, left:0.01, right:0.01},\n  svg={w:0.8, h:0.6}, // percent of container space for svg\n  axes={y:0.1, x:0.1}, // percent of container space for axes,\n  leg={x:0, margin:0, pos:'left'} // absolute width of legend and space on either size\n)\n{\n  if (container == undefined) {container = {w:window.innerWidth, h:window.Height}}\n  // SVG width and height\n\n  var svgSpace =  {\n    w: container.w * svg.w,\n    h: container.h * svg.h\n  }\n\n  var margPx = {\n    top: margins.top * svgSpace.h,\n    bottom: margins.bottom * svgSpace.h,\n    left: margins.left * svgSpace.w,\n    right: margins.right * svgSpace.w\n  },\n\n\n\n  // Space after removing margins\n  chartSpace = {\n    w: svgSpace.w - margPx.left - margPx.right,\n    h: svgSpace.h - margPx.top - margPx.bottom\n  },\n\n  // main dimension of x and y axies\n  // e.g. defines how tall x axis is as length is determined by plotRect.w\n  axesSpace = {\n    x: chartSpace.h * axes.x,\n    y: chartSpace.w * axes.y\n  },\n\n  // space left for drawing the chart properly (e.g. bars, violins, etc)\n  drawingSpace = {\n    x: chartSpace.w - axesSpace.y - leg.x - 2*leg.margin,\n    y: chartSpace.h - axesSpace.x\n  },\n\n\n  legRect = {\n    x: leg.margin + margPx.left + (leg.pos == 'left' ? 0 : drawingSpace.x + axesSpace.y),\n    y: margPx.top, // this is soomehow getting calculated incorectly\n    w: leg.x,\n    h: drawingSpace.y\n  },\n\n  yAxisRect = {\n    x: axesSpace.y + margPx.left + (leg.pos == 'left' ? leg.x + 2*leg.margin : 0),\n    y: margPx.top,\n    w: axesSpace.y,\n    h: drawingSpace.y\n  },\n\n  plotRect = {\n    x: axesSpace.y + margPx.left + (leg.pos == 'left' ? leg.x + 2*leg.margin : 0),\n    y: margPx.top,\n    w: drawingSpace.x,\n    h: drawingSpace.y\n  },\n\n  xAxisRect = {\n    x: axesSpace.y + margPx.left + (leg.pos == 'left' ? leg.x + 2*leg.margin : 0),\n    y: margPx.top + drawingSpace.y,\n    w: drawingSpace.x,\n    h: axesSpace.x\n  }\n\n\n\n  container = d3sm.safeSelect(selection, 'svg', namespace)\n    .style('width', svgSpace.w+'px')\n    .style('height', svgSpace.h+'px')\n\n  var axes = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'axes'))\n\n  var leg = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'legend'))\n  .attr('transform', \"translate(\"+legRect.x+\",\"+legRect.y+\")\")\n\n  var plot = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'plot'))\n    .attr('transform', \"translate(\"+plotRect.x+\",\"+plotRect.y+\")\")\n\n  var xAxis = d3sm.safeSelect(axes, 'g', d3sm.hypenate(namespace, 'x-axis'))\n    .attr('transform', \"translate(\"+xAxisRect.x+\",\"+xAxisRect.y+\")\")\n\n  var yAxis = d3sm.safeSelect(axes, 'g', d3sm.hypenate(namespace, 'y-axis'))\n    .attr('transform', \"translate(\"+yAxisRect.x+\",\"+yAxisRect.y+\")\")\n\n  return {\n    svg: {\n      selection: container,\n      rect: svgSpace\n    },\n    plot: {\n      selection: plot,\n      rect: plotRect\n    },\n    xAxis: {\n      selection: xAxis,\n      rect: xAxisRect\n    },\n    yAxis: {\n      selection: yAxis,\n      rect: yAxisRect\n    },\n    legend: {\n      selection: leg,\n      rect: legRect\n    }\n  }\n}\n\n\n\n/**\n* Adds a clip-path rect and binds it to container\n* @param {d3.selection} container in which to add the clip-path and to which to bind the cliping path to\n* @param {Object} rect the coordinates (x, y, width, height) of the clip-path\n* @param {string} namespace\n* @returns {d3.selection} of the clip-path rect\n*/\nexport function cpRect(container, rect, namespace) {\n  var defs = safeSelect(container, 'defs', hypenate(namespace, 'definitions'))\n  var cp = safeSelect(defs, 'clipPath', hypenate(namespace, 'clip-path'))\n  .attr('id', hypenate(namespace, 'clip-path'))\n\n  var cpRect = safeSelect(cp, 'rect')\n  .attr('x', rect.x)\n  .attr('y', rect.y)\n  .attr('width', rect.width)\n  .attr('height', rect.height)\n\n  defs.raise()\n  // set clipping path to container\n  container.attr('clip-path', 'url(#'+ hypenate(namespace, 'clip-path')+')')\n\n  return cpRect\n}\n\n\n/**\n* Adds a background rect t to container\n* @param {d3.selection} container in which to add the background rectangle\n* @param {Object} rect the coordinates (x, y, width, height) of the background\n* @param {string} fill the color of the background\n* @returns {d3.selection} of the background fill\n*/\nexport function bgRect(container, rect, fill) {\n  return safeSelect(container, 'rect', 'bg')\n  .attr('x', rect.x)\n  .attr('y', rect.y)\n  .attr('width', rect.width)\n  .attr('height', rect.height)\n  .attr('fill', fill)\n}\n\n\n/**\n* Sets up the container for making chart elements. This includes making\n* a clip-path rect bound to the passed container, a background rect, and\n* a g element with class <namespace>-object-container.\n* @param {d3.selection} container in which to add the clip-path and background\n* @param {string} namespace\n* @param {Object} rect the coordinates (x, y, width, height) of the background and clip-path\n* @param {string} fill the color of the background\n* @returns {d3.selection} of g.<namespace>-object-container\n*\n* @see{@link bgRect}\n* @see{@link cpRect}\n*/\nexport function setupContainer(selection, namespace, rect, fill) {\n  // the container for three main items, bg, defs, and object-container\n  var\n  container = safeSelect(selection, 'g', namespace),\n  bg = bgRect(container, rect, fill),\n  cp = cpRect(container, rect, namespace),\n  objectContainer = safeSelect(container, 'g', hypenate(namespace, 'object-container'))\n  return objectContainer\n}\n\n\n/**\n* determines the width of an object for the calling plotting function\n* @param {number} freeSpace how much space is avalible\n* @param {number} numberOfObjects how many object do we need\n* @param {number} minObjectWidth how small are these objects allowed to be\n* @param {number} maxObjectWidth how large are these object allowed to be\n* @param {number} sizeOfSpacer percent of freeSpace that a single spacer should take up (need numberOfObjects - 1 spacers)\n* @param {boolean} overflowQ can we go beyond alloted space\n* @returns {number} how large object should be\n* function tries to keep object within min / max width, but wil default to\n* 5e-10 (smallest consistenly visible by svg size of element) if overflowQ is false\n*/\nexport function calculateWidthOfObject(freeSpace, numberOfObjects, minObjectWidth, maxObjectWidth, sizeOfSpacer, overflowQ) {\n  var sizeOfSpacer =\n  sizeOfSpacer == 0 || sizeOfSpacer > 1\n  ? sizeOfSpacer\n  : freeSpace * sizeOfSpacer\n\n  var numberOfSpacers = numberOfObjects - 1\n  var spaceTakenBySpacers = numberOfSpacers * sizeOfSpacer\n  var remainingSpace = freeSpace - spaceTakenBySpacers\n  remainingSpace = remainingSpace < 0 ? 0 : remainingSpace\n  var objectWidth = remainingSpace / numberOfObjects\n\n  if ( overflowQ && minObjectWidth != undefined && objectWidth < minObjectWidth ) { objectWidth = minObjectWidth }\n  // if ( maxObjectWidth != undefined && objectWidth > maxObjectWidth ) { objectWidth = maxObjectWidth }\n  if ( overflowQ && maxObjectWidth != undefined && objectWidth < maxObjectWidth ) { objectWidth = maxObjectWidth }\n  return Math.max(objectWidth, 5e-10)\n}\n\n/**\n* @param {Array[]} data list data (can be nested). If nested will create more complex spacer size\n* @param {number} freeSpace how much space is avalible\n* @param {number} objectWidth @see{@link calculateWidthOfObject}\n* @param {number} numberOfObjects how many object do we need\n* @param {number} baseSpacerSize percent of freeSpace that a single spacer should take up (need numberOfObjects - 1 spacers)\n* @param {boolean} overflowQ can we go beyond alloted space\n* @returns {number} returns size that spacer should be at level=0\n*/\nexport function calculateWidthOfSpacer(data, freeSpace, objectWidth, numberOfObjects, baseSpacerSize, overflowQ) {\n  if (overflowQ) {\n    // var limitedNumberOfObjects = numberOfObjects > 6 ? 6 : numberOfObjects\n    // var spaceLeft = freeSpace - limitedNumberOfObjects * objectWidth\n    // return spaceLeft / (limitedNumberOfObjects - 1)\n    return freeSpace * baseSpacerSize\n  }\n  var spacersAtEachLevel = spacersNeededAtEachLevel(data)\n  var totalSpacerPercent = total(spacersAtEachLevel.map(function(e, i) {return e * 1 / (i+1)}))\n  var baseSpacerSize = (freeSpace - (objectWidth * numberOfObjects)) / totalSpacerPercent\n  // console.log(freeSpace, objectWidth, numberOfObjects, totalSpacerPercent)\n  // console.log(totalSpacerPercent, baseSpacerSize, totalSpacerPercent * baseSpacerSize)\n  return isNaN(baseSpacerSize) ? 0 : baseSpacerSize\n}\n\n\n/**\n* Calculates number of spacers needed to seperate elements at each level.\n* @param {Array[]} array list data (can be nested). If nested will create more complex spacer size\n* @param {number} [level=0] current level, used in recusrion\n* @param {Array} [levelData=[]] how many spacers needed at a given level\n* @returns {Array} levelData\n*\n* @example\n* array = [[1,2], [3,4]]\n* // returns [1, 2]\n* as at level=0 the only spacer needed is between [1,2] and [3,4]\n* and at level=1 the only two spacers needed is between 1 and 2 as well as\n* 3 and 4 since the spacer between 2 and 3 is handled at level=0\n*/\nexport function spacersNeededAtEachLevel (array, level, levelData ) {\n  if ( level == undefined ) { level = 0;  } else { level += 1 }\n  if ( levelData == undefined ) { levelData = []; }\n  if ( level >= levelData.length ) { levelData.push(array.length - 1) }\n  else { levelData[level] += array.length - 1 }\n  array.map(function(e, i) { if (Array.isArray(e)) { spacersNeededAtEachLevel(e, level, levelData) } } )\n  return levelData\n}\n\n\n\n\n/**\n* Draws a whisker for @see{@link boxwhisker}\n* @param {boolean} dir direction to draw whisker, should be either true (up, top) or false (down or bottom)\n* @param {number} x starting x coordinate in which to draw whisker\n* @param {number} y starting y coordinate in which to draw whisker\n* @param {number} w width of space in which to draw whisker\n* @param {number} h height of space in which to draw whisker\n* @param {number} per percentage of w or h (depends on o) to make whisker\n* @param {boolean} o orientation, true is horizontal and false is vertical\n* @returns {string} representing the svg path (i.e. the d attribute for a path tag)\n*/\nexport function whiskerPath(dir, x, y, w, h, per, o) {\n  // d = direction (true is up), p = percent width\n  if (dir == 'up' || dir == 'top' || dir == true) {dir = true}\n  if (dir == 'down' || dir == 'bottom' || dir == false) {dir = false}\n  o = o == undefined ? 'horizontal' : o\n  per = per == undefined ? 1 : per\n  if (o != \"horizontal\") {\n    var hh = h * per ,\n    w = dir ? w : -w ,\n    a = dir ? x + w : x ,\n    b = dir ? x : x + w ,\n    c = dir ? a : b\n    p = \"M \" + a + ' ' + (     h / 2      ) + ' '\n      + 'L ' + b + ' ' + (     h / 2      ) + ' '\n      + 'M ' + c + ' ' + ( h / 2 - hh / 2 ) + ' '\n      + 'L ' + c + ' ' + ( h / 2 + hh / 2 ) + ' '\n\n    return p\n  }\n  var ww = w * per,\n  a = dir ? y + h : y  ,\n  b = dir ? y : y + h  ,\n  p = \"M \" + (  w / 2  ) + ' ' + a + ' ' // straight line part\n    + 'L ' + (  w / 2  ) + ' ' + b + ' ' // straight line part\n    + 'h ' + ( -ww / 2 ) + ' ' + 0 + ' ' // horizontal line part\n    + 'h ' + (    ww   ) + ' ' + 0 + ' '\n  return p\n}\n\n\n\n\n\n\n\n\n\n\n\n\n\nexport function resizeDebounce(f, wait) {\n  var resize = debounce(function(){f()},wait)\n  window.addEventListener('resize', resize)\n}\n\n\n\nfunction debounce(func, wait, immediate) {\n  var timeout;\n    return function() {\n        var context = this, args = arguments;\n        var later = function() {\n            timeout = null;\n            if (!immediate) func.apply(context, args);\n        };\n        var callNow = immediate && !timeout;\n        clearTimeout(timeout);\n        timeout = setTimeout(later, wait);\n        if (callNow) func.apply(context, args);\n    };\n}\n","import {log, warn, error, info} from './utils';\n/*******************************************************************************\n**                                                                            **\n**                                                                            **\n**                              SPACEGROUPING                                 **\n**                                                                            **\n**                                                                            **\n*******************************************************************************/\n/**\n * Produces a function for spacing objects by an arbitrarly complex grouping\n * @returns {recursivelyPosition} the function for moving the objects\n * (see {@link groupingSpacer#recursivelyPosition})\n * @namespace groupingSpacer\n */\nexport function groupingSpacer() {\n  var\n  /*@var {boolean} horizontalQ @default*/\n\n  /**\n  * Whether or not to space objects horizontally or vertically.\n  * (see {@link groupingSpacer.horizontalQ})\n  * @param {boolean} [horizontalQ=true]\n  * @memberof groupingSpacer#\n  * @instance\n  */\n  horizontalQ = true,\n  /**\n  * The scale to use to position elements if {@link groupingSpacer#moveby}=\"string\"\n  * (see {@link groupingSpacer.scale})\n  * @param {d3.scale} [scale=d3.scaleLinear()]\n  * @memberof groupingSpacer#\n  * @instance\n  */\n  scale = d3.scaleLinear(),\n  /**\n  * How elements in the complex grouping should be moved over by.\n  * By default, moveby=\"category\", which moves objects by the complex grouping\n  * But objects can also be moved over by scale.\n  * (see {@link groupingSpacer.moveby})\n  * @param {string} [moveby=\"category\"]\n  * @memberof groupingSpacer#\n  * @instance\n  */\n  moveby = 'category',\n  /**\n  * How many objects are there in total\n  * (see {@link groupingSpacer.numberOfObjects})\n  * @param {number} [numberOfObjects=none]\n  * @memberof groupingSpacer#\n  * @instance\n  */\n  numberOfObjects,\n  /**\n  * The class given to an nested <g> tag whose parent(s) have the correct transition\n  * properties\n  * (see {@link groupingSpacer.numberOfObjects})\n  * @param {string} [numberOfObjects='d3sm-groupped-item']\n  * @memberof groupingSpacer#\n  * @instance\n  */\n  objectClass = 'd3sm-groupped-item',\n  /**\n  * The size of the objects being positioned\n  * (see {@link groupingSpacer.objectSize})\n  * @param {number} [objectSize=none]\n  * @memberof groupingSpacer#\n  * @instance\n  */\n  objectSize,\n  /**\n  * The size of the un-nested spacer between objects\n  * (see {@link groupingSpacer.spacerSize})\n  * @param {number} [spacerSize=none]\n  * @memberof groupingSpacer#\n  * @instance\n  */\n  spacerSize,\n  /**\n  * The duration of transitions in ms\n  * (see {@link groupingSpacer.transitionDuration})\n  * @param {number} [transitionDuration=1000]\n  * @memberof groupingSpacer#\n  * @instance\n  */\n  transitionDuration = 1000,\n  /**\n  * The ease function for the transitions\n  * (see {@link groupingSpacer.easeFunc})\n  * @param {d3.ease} [easeFunc=d3.easeSin]\n  * @memberof groupingSpacer#\n  * @instance\n  */\n  easeFunc = d3.easeSin,\n  /**\n  * The namespace for the objects being moved\n  * (see {@link groupingSpacer.namespace})\n  * @param {string} [namespace='spacer']\n  * @memberof groupingSpacer#\n  * @instance\n  */\n  namespace = 'spacer',\n  /**\n  * The animation for new objects being added\n  * (see {@link groupingSpacer.enterFunction})\n  * @param {function} enterFunction\n  * @memberof groupingSpacer#\n  * @instance\n  * @example\n  * // by default\n  * function(newObjectSelection) {\n  *  newObjectSelection.attr('transform', function(d, i){\n  *    var\n  *    x = horizontalQ ? objectSize * numberOfObjects + spacerSize * (numberOfObjects - 1) : 0,\n  *    y = !horizontalQ ? objectSize * numberOfObjects + spacerSize * (numberOfObjects - 1) : 0,\n  *    t = 'translate('+x+','+y+')'\n  *    return t\n  *  })\n  * }\n  */\n  enterFunction = function(cur) {\n    cur.attr('transform', function(d, i){\n      var\n      // x = horizontalQ ? objectSize * numberOfObjects + spacerSize * (numberOfObjects - 1) : 0,\n      // y = !horizontalQ ? objectSize * numberOfObjects + spacerSize * (numberOfObjects - 1) : 0,\n      x = horizontalQ ? window.outerWidth : 0,\n      y = !horizontalQ ? window.outerWidth : 0,\n      t = 'translate('+x+','+y+')'\n      // if(y == undefined) {console.log(cur.node(), y, d)}\n      return t\n    })\n  },\n  /**\n  * The animation for old objects being removed\n  * (see {@link groupingSpacer.exitFunction})\n  * @param {function} exitFunction\n  * @memberof groupingSpacer#\n  * @instance\n  * @example\n  * // by default\n  * oldObjectSelection.transition().duration(transitionDuration).ease(easeFunc)\n  * .attr('transform', function(d, i){\n  *     var\n  *   x = horizontalQ ? objectSize * numberOfObjects + spacerSize * (numberOfObjects - 1) : 0,\n  *   y = !horizontalQ ? objectSize * numberOfObjects + spacerSize * (numberOfObjects - 1) : 0,\n  *   t = 'translate('+x+','+y+')'\n  *   return t\n  * }).remove()\n  */\n  exitFunction = function(cur){\n    log(\"groupingSpacer\", \"exiting with\", {current: cur, currentNode: cur.node()})\n    cur.selectAll('g').classed('to-remove', true)\n\n    cur.transition().duration(transitionDuration*0.9).ease(easeFunc)\n    .attr('transform', function(d, i){\n      var\n      // x = horizontalQ ? objectSize * numberOfObjects + spacerSize * (numberOfObjects - 1) : 0,\n      // y = !horizontalQ ? objectSize * numberOfObjects + spacerSize * (numberOfObjects - 1) : 0,\n      x = horizontalQ ? window.outerWidth : 0,\n      y = !horizontalQ ? window.outerWidth : 0,\n      t = 'translate('+x+','+y+')'\n      // if(y == undefined) {console.log(cur.node(), y, d)}\n      return t\n    }).remove()\n  }\n\n  /**\n   * Gets / sets horizontalQ (whether or not to space objects horizontally or vertically).\n   * (see {@link groupingSpacer#horizontalQ})\n   * @param {string} [_=none]\n   * @returns {groupingSpacer | string}\n   * @memberof groupingSpacer\n   * @static\n   */\n  recursivelyPosition.horizontalQ = function(_) { return arguments.length ? (horizontalQ = _, recursivelyPosition) : horizontalQ }\n  /**\n   * Gets / sets the scale to use to position elements if {@link groupingSpacer#moveby}=\"string\"\n   * (see {@link groupingSpacer#scale})\n   * @param {d3.scale} [_=none]\n   * @returns {groupingSpacer | d3.scale}\n   * @memberof groupingSpacer\n   * @static\n   */\n  recursivelyPosition.scale = function(_) { return arguments.length ? (scale = _, recursivelyPosition) : scale }\n  /**\n   * Gets / sets moveby (whether or not to move by scale or by grouping).\n   * (see {@link groupingSpacer#moveby})\n   * @param {string} [_=none]\n   * @returns {groupingSpacer | string}\n   * @memberof groupingSpacer\n   * @static\n   */\n  recursivelyPosition.moveby = function(_) { return arguments.length ? (moveby = _, recursivelyPosition) : moveby }\n  /**\n   * Gets / sets numberOfObjects.\n   * (see {@link groupingSpacer#numberOfObjects})\n   * @param {number} [_=none]\n   * @returns {groupingSpacer | number}\n   * @memberof groupingSpacer\n   * @static\n   */\n  recursivelyPosition.numberOfObjects = function(_) { return arguments.length ? (numberOfObjects = _, recursivelyPosition) : numberOfObjects }\n  /**\n   * Gets / sets the objectClass (will be applied to <g> elements).\n   * (see {@link groupingSpacer#objectClass})\n   * @param {string} [_=none]\n   * @returns {groupingSpacer | string}\n   * @memberof groupingSpacer\n   * @static\n   */\n  recursivelyPosition.objectClass = function(_) { return arguments.length ? (objectClass = _, recursivelyPosition) : objectClass }\n  /**\n   * Gets / sets the objectSize.\n   * (see {@link groupingSpacer#objectSize})\n   * @param {number} [_=none]\n   * @returns {groupingSpacer | number}\n   * @memberof groupingSpacer\n   * @static\n   */\n  recursivelyPosition.objectSize = function(_) { return arguments.length ? (objectSize = _, recursivelyPosition) : objectSize }\n  /**\n   * Gets / sets the spacerSize.\n   * (see {@link groupingSpacer#spacerSize})\n   * @param {number} [_=none]\n   * @returns {groupingSpacer | number}\n   * @memberof groupingSpacer\n   * @static\n   */\n  recursivelyPosition.spacerSize = function(_) { return arguments.length ? (spacerSize = _, recursivelyPosition) : spacerSize }\n  /**\n   * Gets / sets the transitionDuration.\n   * (see {@link groupingSpacer#transitionDuration})\n   * @param {number} [_=none]\n   * @returns {groupingSpacer | number}\n   * @memberof groupingSpacer\n   * @static\n   */\n  recursivelyPosition.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, recursivelyPosition) : transitionDuration }\n  /**\n   * Gets / sets the easeFunc.\n   * (see {@link groupingSpacer#easeFunc})\n   * @param {d3.ease} [_=none]\n   * @returns {groupingSpacer | d3.ease}\n   * @memberof groupingSpacer\n   * @static\n   */\n  recursivelyPosition.easeFunc = function(_) { return arguments.length ? (easeFunc = _, recursivelyPosition) : easeFunc }\n  /**\n   * Gets / sets the namespace.\n   * (see {@link groupingSpacer#namespace})\n   * @param {string} [_=none]\n   * @returns {groupingSpacer | string}\n   * @memberof groupingSpacer\n   * @static\n   */\n  recursivelyPosition.namespace = function(_) { return arguments.length ? (namespace = _, recursivelyPosition) : namespace }\n  /**\n   * Gets / sets the enterFunction.\n   * (see {@link groupingSpacer#enterFunction})\n   * @param {function} [_=none]\n   * @returns {groupingSpacer | function}\n   * @memberof groupingSpacer\n   * @static\n   */\n  recursivelyPosition.enterFunction = function(_) { return arguments.length ? (enterFunction = _, recursivelyPosition) : enterFunction }\n  /**\n   * Gets / sets the exitFunction.\n   * (see {@link groupingSpacer#exitFunction})\n   * @param {function} [_=none]\n   * @returns {groupingSpacer | function}\n   * @memberof groupingSpacer\n   * @static\n   */\n  recursivelyPosition.exitFunction = function(_) { return arguments.length ? (exitFunction = _, recursivelyPosition) : exitFunction }\n\n\n  /**\n   * recursively position the objects inside of the selection.\n   * @param {d3.selection} selection\n   * @param {Object} data\n   * @param {level} [level=0] recursion depth\n   * @returns {number} (how much to move next element)\n   * @memberof groupingSpacer#\n   */\n  function recursivelyPosition(selection, data, level) {\n    if ( level == undefined ) { level = 0;  }\n\n    var currentSelection = selection.selectAll('g.'+namespace+'[level=\"'+level+'\"]').data(data)\n    var enter = currentSelection.enter().append('g').attr('level', level).attr('class', namespace)\n    var exit = currentSelection.exit()\n    currentSelection = currentSelection.merge(enter)\n\n\n    if (typeof exitFunction == 'function' ){ exit.each(function(d, i){ exitFunction(d3.select(this))}) }\n    else{exit.remove()}\n    // spacer for current level\n    var levelSpacer = spacerSize / (level+1)\n    // movement for current level\n    var move = 0\n    currentSelection.each(function(currentElement, index) {\n      var t = d3.select(this)\n      if (t.attr('transform') == undefined && typeof enterFunction == 'function') { enterFunction(t) }\n\n      t.transition().duration(transitionDuration).ease(easeFunc)\n      .attr('transform', function(d, i) {\n        var\n        x = horizontalQ ? (moveby ==\"scale\" ? scale(d) : move) : 0,\n        y = !horizontalQ ? (moveby ==\"scale\" ? scale(d) : move): 0,\n        t = 'translate('+x+','+y+')'\n        return t\n      })\n\n      if (Array.isArray(currentElement)) {\n        move += recursivelyPosition(t, currentElement, level+1)\n        var toRemove = t.selectAll('g.'+namespace+'[level=\"'+(level)+'\"] > g.'+objectClass+'.'+namespace)\n        if (typeof exitFunction == 'function' ){ toRemove.each(function(d, i){ exitFunction(d3.select(this))}) }\n        else{toRemove.remove()}\n      }\n      else {\n        move += objectSize\n        var obj = t.select('g.'+namespace+'[level=\"'+level+'\"] > g.'+objectClass+'.'+namespace)\n        if (obj.empty()) { obj = t.append('g').attr('class', objectClass).classed(namespace, true) }\n        obj.attr('parent-index', index)\n        var toRemove = t.selectAll('g.'+namespace+'[level=\"'+(level+1)+'\"]')\n\n        if (typeof exitFunction == 'function' ){ toRemove.each(function(d, i){ exitFunction(d3.select(this))}) }\n        else{toRemove.remove()}\n      }\n      move += (index == currentSelection.size()-1) ? 0 : levelSpacer\n    })\n    return move\n  }\n  return recursivelyPosition\n}\n","import {modifyHexidecimalColorLuminance} from './helpers';\n\n/**\n * Creates a colorFunction\n * @constructor colorFunction\n * @namespace colorFunction\n * @returns {function} colorFunction\n */\nexport function colorFunction() {\n  var\n  data,\n\n  /**\n  * Default colors to use\n  * @param {number[]} [colors=[\"#2c7bb6\", \"#00a6ca\", \"#00ccbc\", \"#90eb9d\", \"#ffff8c\", \"#f9d057\", \"#f29e2e\", \"#e76818\", \"#d7191c\"]]\n  * @memberof colorFunction#\n  * @property\n  */\n  colors = [\"#2c7bb6\", \"#00a6ca\", \"#00ccbc\", \"#90eb9d\", \"#ffff8c\", \"#f9d057\", \"#f29e2e\", \"#e76818\", \"#d7191c\"],\n  /**\n  * Interpolator for colors\n  * @param {d3.interpolation} [interpolation=d3.interpolateRgb]\n  * @memberof colorFunction#\n  * @property\n  */\n  interpolation = d3.interpolateRgb,\n  /**\n  * Function for modifying color luminance\n  * @param {function} [modifyOpacity=modifyHexidecimalColorLuminance]\n  * @memberof colorFunction#\n  * @property\n  */\n  modifyOpacity = modifyHexidecimalColorLuminance,\n  /**\n  * How to modify color for stroke\n  * @param {number} [strokeOpacity=0]\n  * @memberof colorFunction#\n  * @property\n  */\n  strokeOpacity = 0,\n  /**\n  * How to modify color for fill\n  * @param {number} [fillOpacity=0.4]\n  * @memberof colorFunction#\n  * @property\n  */\n  fillOpacity = 0.4,\n  /**\n  * How to determine the color to use\n  * @param {string} [colorBy='index']\n  * @memberof colorFunction#\n  * @property\n  */\n  colorBy = 'index',\n  /**\n  * Sets the scale for interpolating the colors\n  * @param {number[]} [dataExtent=[0, colors.length - 1]]\n  * @memberof colorFunction#\n  * @property\n  */\n  dataExtent = [0, colors.length - 1],\n  /**\n  * Extracts the value to color by\n  * @param {function} [valueExtractor=function(k, v, i) {return v}]\n  * @memberof colorFunction#\n  * @property\n  */\n  valueExtractor = function(k, v, i) {return v},\n\n  /**\n  * Extracts the category to color by\n  * @param {function} [categoryExtractor=function(k, v, i) {return v.category}]\n  * @memberof colorFunction#\n  * @property\n  */\n  categoryExtractor = function(k, v, i) {return v.category},\n\n  /**\n  * The different type of categories of which to color by\n  * @param {string[]} [categories=undefined]\n  * @memberof colorFunction#\n  * @property\n  */\n  categories,\n\n  /**\n  * Scale for interpolating the colors\n  * @param {d3.scale} [scale=d3.scaleLinear()]\n  * @memberof colorFunction#\n  * @property\n  */\n  scale = d3.scaleLinear()\n  .interpolate(interpolation).domain(dataExtent).range(colors),\n  helperScale = d3.scaleLinear()\n\n\n\n  // var h = x => '#' + x.match(/\\d+/g).map(y = z => ((+z < 16)?'0':'') + (+z).toString(16)).join('');\n  var h = function(x) {\n    return \"#\" + x.match(/\\d+/g).map(\n      function(y, i) {\n        return  ((+y < 16)?'0':'') + (+y).toString(16)\n      }).join('');\n  }\n\n  /**\n   * Gets or sets the default colors\n   * (see {@link colorFunction#colors})\n   * @param {number[]} [_=none]\n   * @returns {colorFunction | number[]}\n   * @memberof colorFunction\n   * @property\n   */\n  colorFunction.colors = function(_) {\n    return arguments.length\n    ?\n      (\n        colors = _,\n        scale.range(colors),\n        colorFunction\n      )\n    : colors;\n  };\n  /**\n   * Gets or sets the function for interpolating the colors\n   * (see {@link colorFunction#interpolation})\n   * @param {d3.interpolation} [_=none]\n   * @returns {colorFunction | d3.interpolation}\n   * @memberof colorFunction\n   * @property\n   */\n  colorFunction.interpolation = function(_) {\n    return arguments.length\n    ?\n    (\n      interpolation = _,\n      scale.interpolate(interpolation).range(colors),\n      colorFunction\n    )\n    : interpolation;\n  };\n  /**\n   * Gets or sets the values for the scale which transforms the value to a color\n   * (see {@link colorFunction#dataExtent})\n   * @param {number[]} [_=none]\n   * @returns {colorFunction | number[]}\n   * @memberof colorFunction\n   * @property\n   */\n  colorFunction.dataExtent = function(_) {\n    return arguments.length\n    ? (\n        dataExtent = _,\n        scale.domain(dataExtent).interpolate(scale.interpolate()),\n        colorFunction\n      )\n    : dataExtent;\n  };\n  /**\n   * Gets or sets the vthe scale which transforms the value to a color\n   * (see {@link colorFunction#scale})\n   * @param {d3.scale} [_=none]\n   * @returns {colorFunction | d3.scale}\n   * @memberof colorFunction\n   * @property\n   */\n  colorFunction.scale = function(_) {\n    return arguments.length\n    ? (\n        _ = _.domain(scale.domain()).interpolate(scale.interpolate()).range(scale.range()),\n        scale = _,\n        colorFunction\n      )\n    : scale;\n  };\n  /**\n   * Gets or sets the function for modify opacity\n   * (see {@link colorFunction#modifyOpacity})\n   * @param {function} [_=none]\n   * @returns {colorFunction | function}\n   * @memberof colorFunction\n   * @property\n   */\n  colorFunction.modifyOpacity = function(_) { return arguments.length ? (modifyOpacity = _, colorFunction) : modifyOpacity; };\n  /**\n   * Gets or sets the value to modify the color for the stroke via {@link colorFunction#modifyOpacity}\n   * (see {@link colorFunction#strokeOpacity})\n   * @param {number} [_=none]\n   * @returns {colorFunction | number}\n   * @memberof colorFunction\n   * @property\n   */\n  colorFunction.strokeOpacity = function(_) { return arguments.length ? (strokeOpacity = _, colorFunction) : strokeOpacity; };\n  /**\n   * Gets or sets the value to modify the color for the stroke via {@link colorFunction#fillOpacity}\n   * (see {@link colorFunction#fillOpacity})\n   * @param {number} [_=none]\n   * @returns {colorFunction | number}\n   * @memberof colorFunction\n   * @property\n   */\n  colorFunction.fillOpacity = function(_) { return arguments.length ? (fillOpacity = _, colorFunction) : fillOpacity; };\n  /**\n   * Gets or sets the value to colorBy\n   * (see {@link colorFunction#colorBy})\n   * @param {string} [_=none]\n   * @returns {colorFunction | string}\n   * @memberof colorFunction\n   * @property\n   */\n  colorFunction.colorBy = function(_) { return arguments.length ? (colorBy = _, colorFunction) : colorBy; };\n  /**\n   * Gets or sets the value of valueExtractor\n   * (see {@link colorFunction#valueExtractor})\n   * @param {function} [_=none]\n   * @returns {colorFunction | function}\n   * @memberof colorFunction\n   * @property\n   */\n  colorFunction.valueExtractor = function(_) { return arguments.length ? (valueExtractor = _, colorFunction) : valueExtractor; };\n\n  /**\n   * Gets or sets the value of categoryExtractor\n   * (see {@link colorFunction#categoryExtractor})\n   * @param {function} [_=none]\n   * @returns {colorFunction | function}\n   * @memberof colorFunction\n   * @property\n   */\n  colorFunction.categoryExtractor = function(_) { return arguments.length ? (categoryExtractor = _, colorFunction) : categoryExtractor; };\n  /**\n   * Gets or sets the value of categoryExtractor\n   * (see {@link colorFunction#categories})\n   * @param {string[]} [_=none]\n   * @returns {colorFunction | string[]}\n   * @memberof colorFunction\n   * @property\n   */\n  colorFunction.categories = function(_) { return arguments.length ? (categories = _, colorFunction) : categories; };\n\n\n  function colorFunction(key, value, index, type, hoverQ) {\n    var c,\n    opac = type == \"fill\" ? fillOpacity : strokeOpacity;\n\n\n    updateScale()\n\n    if (colorBy == \"index\") {\n      c = (type != undefined) ? modifyOpacity(h(scale(index)), opac) : h(scale(index))\n    }\n\n    else if (colorBy == 'value') {\n      var v = valueExtractor(key, value, index);\n      // if (v < dataExtent[0]) {dataExtent[0] = v; updateScale()}\n      // if (v > dataExtent[1]) {dataExtent[1] = v; updateScale()}\n\n      c = (type != undefined) ? modifyOpacity(h(scale(v)), opac) : h(scale(v))\n    }\n\n    else if (colorBy == 'category' ){\n      var cat = categoryExtractor(key, value, index);\n      var v = categories.indexOf(cat)\n      c = (type != undefined) ? modifyOpacity(h(scale(v)), opac) : h(scale(v))\n\n    }\n\n    else {\n      c = (type != undefined) ? modifyOpacity(h(scale(index)), opac) : h(scale(index))\n    }\n\n    return c\n  }\n\n  function updateScale(){\n\n\n    helperScale.domain([0, colors.length])\n    if (colorBy == 'category' && categories != undefined) { helperScale.range([0, categories.length]) }\n    else { helperScale.range(dataExtent) }\n\n\n    var a = Array(colors.length).fill(0).map(function(d, i){ return helperScale(i) })\n    scale.domain(a)\n  }\n\n  return colorFunction\n}\n","import {safeSelect, round} from './helpers';\nimport {log, warn, info, error, consoleGroup, consoleGroupEnd} from './utils';\n/*******************************************************************************\n**                                                                            **\n**                                                                            **\n**                                 TOOLTIP                                    **\n**                                                                            **\n**                                                                            **\n*******************************************************************************/\n/**\n * Produces a function for handling the tooltip\n *\n * {@link https://sumneuron.gitlab.io/d3sm/demos/tooltip-design/index.html Demo}\n * @param {d3.selection} selection\n * @returns {tooltip}\n * @namespace tooltip\n */\nexport function tooltip( selection ) {\n\n  var\n  keys,\n  values,\n  header,\n  data,\n  selection\n\n  /**\n   * Gets / sets the keys to be displayed in the tooltip.\n   * If not set, uses d3.keys(data[key])\n   * @param {string[]} [_=none]\n   * @returns {tooltip | string[]}\n   * @memberof tooltip\n   */\n  tooltip.keys = function(_){return arguments.length ? (keys = _, tooltip) : keys};\n  /**\n   * Gets / sets the values to be displayed next to the keys.\n   * If not set, uses data[key][keys[i]].\n   * If a function, gets passed currentData (data[key]) and keys[i].\n   * @param {*[]} [_=none]\n   * @returns {tooltip | *[]}\n   * @memberof tooltip\n   */\n  tooltip.values = function(_){return arguments.length ? (values = _, tooltip) : values};\n  /**\n   * Gets / sets the header to be displayed in the tooltip.\n   * If not set, uses key\n   * @param {string} [_=none]\n   * @returns {tooltip | string}\n   * @memberof tooltip\n   */\n  tooltip.header = function(_){return arguments.length ? (header = _, tooltip) : header};\n  /**\n   * Gets / sets the data (over the selection) to be used for the tooltip\n   * @param {Object} [_=none]\n   * @returns {tooltip | Object}\n   * @memberof tooltip\n   */\n  tooltip.data = function(_){return arguments.length ? (data = _, tooltip) : data};\n  /**\n   * Gets / sets the selection for the tooltip to be applied on\n   * @param {d3.selection} [_=none]\n   * @returns {tooltip | d3.selection}\n   * @memberof tooltip\n   */\n  tooltip.selection = function(_){return arguments.length ? (selection = _, tooltip) : selection};\n\n  /**\n   * Bind, via selection.on(), the mousemove and mouseout events\n   * @returns undefined\n   */\n  function tooltip( ) {\n    selection.on('mouseover', mousemove)\n    selection.on('mousemove', mousemove)\n    selection.on('mouseout', function(){ d3.selectAll(\".d3sm-tooltip\").remove()})\n  }\n\n\n  /**\n   * Produces the tooltip on mousemove\n   * @param {string} key of the object targeted by the mousemove\n   * @param {number} i (index) of the object targeted by mousemove\n   * @memberof tooltip\n   * @private\n   */\n  function mousemove(key, i) {\n    consoleGroup('d3sm-tooltip')\n    var currentData = data[key]\n\n    var [x, y] = d3.mouse(d3.select(\"html\").node())\n    log('tooltip', 'mousemove detected',{key: key, index: i, x:x, y:y})\n    log('tooltip', 'current data', currentData)\n\n\n\n    var div = safeSelect(d3.select('html'), 'tooltip', 'd3sm-tooltip')\n    .classed('card', true)\n    .style('max-width', '300px')\n    .style('background-color', \"#212529\")\n    .style('color', 'white')\n\n\n\n    var cardBody = safeSelect(div, 'div', 'card-body')\n    var cardTitle = safeSelect(cardBody, 'h5', 'card-title')\n    .text(header == undefined ? key : typeof header == 'function' ? header(key, currentData, i) : header)\n    .style('color', 'cyan')\n\n\n    var table = safeSelect(cardBody, 'table', 'table').classed('table-dark', true)\n    var tBody = safeSelect(table, 'tbody')\n\n    tBody = tBody.selectAll('tr')\n    tBody= tBody.data(keys == undefined ? d3.keys(currentData): keys)\n    tBody.exit().remove()\n\n\n    var tr = tBody.enter().append('tr').style('max-width', '300px')\n    tr.append('td').attr('class', function(d, i){return 'tooltip-key'})\n    tr.append('td').attr('class',  function(d, i, j){return 'tooltip-value'})\n    .attr('tooltip-row-index', function(d, i){return i})\n\n    // tBody = tBody.merge(tr)\n    consoleGroup('tooltip-rows')\n    tBody.selectAll('.tooltip-key').text(function(d, i){return d})\n    tBody.selectAll('tr .tooltip-value')\n    .text(function(d, i){\n      log('tooltip', 'trying to set value', {rowKey: d, rowIndex: i})\n      var i = d3.select(this).attr('tooltip-row-index')\n      var v = currentData[d];\n\n\n      if (values != undefined) {v = values[i]; if(typeof v == \"function\") {v = v(currentData, d)}}\n      return  typeof v == 'number' ? round(v, 5) : v\n    })\n    consoleGroupEnd()\n    consoleGroupEnd()\n\n    x += 15\n    // x += 15\n    var bbox = div.node().getBoundingClientRect()\n    if (x + bbox.width > window.innerWidth - window.scrollX) { x = d3.event.pageX - bbox.width - 15 }\n    if (y + bbox.height > window.innerHeight  - window.scrollY) { y = d3.event.pageY - bbox.height - 15 }\n    div.style('position') == \"relative\"\n    ? div.style('position', 'absolute').style('left', x+'px').style('top', y+'px')\n    : div.style('left', x+'px').style('top', y+'px')\n    // .transition().duration(200).ease(d3.easeSin)\n\n    // if (bbox.x + bbox.width > window.innerWidth) {\n    //   div.style('left', (d3.event.pageX-15-bbox.width)+'px')\n    // }\n    // if (bbox.y + bbox.height > window.innerHeight) {\n    //   div.style('top', (d3.event.pageY-15-bbox.height)+'px')\n    // }\n\n    div.attr('z-index', 10000)\n  }\n\n  return tooltip\n}\n","import {hypenate, safeSelect} from './helpers';\n\nexport function selectFilter(selection) {\n\n  var\n  data,\n  namespace = 'd3sm-select-filter',\n  selectionName = 'Select options:',\n  defaultValue = undefined\n\n\n\n\n  var lastValue = undefined\n\n  selectFilter.data = function(_) { return arguments.length ? (data = _, selectFilter) : data}\n  selectFilter.namespace = function(_) { return arguments.length ? (namespace = _, selectFilter) : namespace}\n  selectFilter.selectionName = function(_) { return arguments.length ? (selectionName = _, selectFilter) : selectionName}\n  selectFilter.defaultValue = function(_) { return arguments.length ? (defaultValue = _, selectFilter) : defaultValue}\n  selectFilter.currentOption = currentOption\n\n  function selectFilter() {\n    var\n    container = safeSelect(selection, 'div', 'input-group').classed(hypenate(namespace,'container'),true),\n\n      selectPrepend = safeSelect(container, 'div', 'select-prepend').classed('input-group-prepend', true),\n        selectPrependSpan = safeSelect(selectPrepend, 'span', 'input-group-text').text(selectionName),\n\n      select = safeSelect(container, 'select', 'custom-select').classed(hypenate(namespace,'select'),true),\n\n      selectAppend = safeSelect(container, 'div', 'select-append').classed('input-group-prepend', true),\n        selectAppendButton = safeSelect(selectAppend, 'a', 'filter-button').classed('btn btn-outline-secondary', true),\n          filterButtonIcon = safeSelect(selectAppendButton, 'i', 'fa fa-filter'),\n\n      inputGroup = safeSelect(container, 'div', 'filter-input-group').classed('input-group',true).classed('d-none', true),\n        inputPrepend = safeSelect(inputGroup, 'div', 'input-group-prepend'),\n          inputPrependSpan = safeSelect(inputPrepend, 'span', 'input-group-text').classed('search-button', true),\n            inputPrependSpanIcon = safeSelect(inputPrependSpan,'i','fa fa-search'),\n\n        input = safeSelect(inputGroup, 'input', 'form-control').attr('placeholder', 'all').attr('type', 'text'),\n        inputAppend = safeSelect(inputGroup, 'div', 'input-group-append'),\n          inputAppendButton = safeSelect(inputAppend, 'a', 'close-button').classed('btn btn-outline-secondary', true),\n            inputAppendButtonIcon = safeSelect(inputAppendButton, 'i', 'fa fa-close')\n\n\n    var keys = d3.keys(data),\n    options = select.selectAll('option')\n\n    options = options.data(d3.keys(data))\n    options = options.merge(options.enter().append('option'))\n    .attr('value', function(d, i){return d})\n    .text(function(d, i){return d})\n\n    var\n    filterButton = selectAppendButton,\n    closeButton = inputAppendButton\n\n    filterButton.on('click', function(d, i){\n      var currentStyle = inputGroup.classed('d-none')\n      inputGroup.classed('d-none', !currentStyle)\n    })\n\n    closeButton.on('click', function(d, i){\n      input.property('value', '').dispatch('input')\n    })\n\n    input.on('input', function(d, i){\n      var\n      val = input.property('value'),\n      reg = new RegExp(val, 'gi'),\n      use\n\n      if (val == '') {use = keys}\n      else {\n        use = []\n        d3.keys(data).map(function(option, j){\n          var match = option.match(reg)\n          if (match == null || match.join('') == '') {}\n          else { use.push(option) }\n        })\n      }\n\n      options = select.selectAll('option')\n      options = options.data(use)\n      options.exit().remove()\n      options = options.merge(options.enter().append('option'))\n      .attr('value', function(d, i){return d})\n      .text(function(d, i){return d})\n\n      var current = currentOption()\n      if (lastValue != current) {\n        lastValue = current\n        select.dispatch('change')\n      }\n    })\n\n\n  }\n\n  function currentOption() {\n    var val = selection.select(\"select\").property('value')\n    return val == undefined || val == ''\n    ? defaultValue == undefined\n      ? d3.keys(data)[0]\n      : defaultValue\n    : val\n  }\n\n  return selectFilter\n}\n","import {hypenate, safeSelect, euclideanDistance} from './helpers';\nimport {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer} from './utils';\nimport {unique, hasQ, flatten, whichBin} from './array-functions';\nimport {groupingSpacer} from './grouping-spacer';\nimport {colorFunction as CF} from './color-function';\nimport {tooltip as TTip} from './tooltip';\nimport './d3-prototypes';\n\nfunction getTranslation(selection){\n  var transform = selection.attr('transform')\n  var [junk, xy] =transform.split('translate(')\n  var [x, y] = xy.split(',')\n  y, junk = y.split(')')\n  return [parseFloat(x), parseFloat(y)]\n}\n\nexport function lasso( selection ) {\n  var\n  svg, // svg that is target of events\n  objectContainer, // container which houses objects we are selecting (allows for transform to be applied to lasso)\n  objectClass, // class of object we are selecting\n  namespace=\"d3sm-lasso\",\n  chartContainer,\n  chartOffset,\n  objectsOffset,\n  eventCatcher,\n\n  xScale, // optional scale for the lasso currentPoints\n  yScale, // optional scale for the lasso currentPoints\n\n  activeQ = false, // whether or not lasso is active\n\n  currentPoints=[], // mouse points for current lasso\n  allPoints=[], // list of lists for all points of lassos\n\n  line = d3.line()\n  .x(function(d, i){\n    var x\n    if (xScale != undefined) { x = xScale(d[0]) }\n    else {x = d[0]}\n    return x //- chartOffset[0]// - objectsOffset[0]\n  })\n  .y(function(d, i){\n    var y\n    if (yScale != undefined) { y= yScale(d[1]) }\n    else {y =  d[1]}\n    return y// - chartOffset[1]// - objectsOffset[1]\n  })\n  .curve(d3.curveLinearClosed),\n\n  instance=0,    // an indentifier for which instance this lasso is under the current svg\n\n  tickDistance = 10,\n\n  // styles for lasso path\n  color = '#17a2b8',\n  animationRate = '10s',\n  opacity=0.3,\n  dashArray = '5, 10',\n  stroke = 'black',\n  strokeWidth=2,\n\n  // styles for lassoed objects\n  lassoedFill = \"white\",\n  lassoedStroke = 'black',\n  lassoedStrokeWidth = 3,\n\n  transitionDuration = 1000,\n  easeFunc = d3.easeExp\n\n  var path\n\n  lasso.svg = function(_) { return arguments.length ? (svg = _, lasso) : svg; }\n  lasso.chartContainer = function(_) { return arguments.length ? (chartContainer = _, lasso) : chartContainer; }\n  lasso.objectContainer = function(_) { return arguments.length ? (objectContainer = _, lasso) : objectContainer; }\n  lasso.objectClass = function(_) { return arguments.length ? (objectClass = _, lasso) : objectClass; }\n  lasso.namespace = function(_) { return arguments.length ? (namespace = _, lasso) : namespace; }\n  lasso.xScale = function(_) { return arguments.length ? (xScale = _, lasso) : xScale; }\n  lasso.yScale = function(_) { return arguments.length ? (yScale = _, lasso) : yScale; }\n  lasso.activeQ = function(_) { return arguments.length ? (activeQ = _, lasso) : activeQ; }\n  lasso.currentPoints = function(_) { return arguments.length ? (currentPoints = _, lasso) : currentPoints; }\n  lasso.allPoints = function(_) { return arguments.length ? (allPoints = _, lasso) : allPoints; }\n  lasso.instance = function(_) { return arguments.length ? (instance = _, lasso) : instance; }\n  lasso.tickDistance = function(_) { return arguments.length ? (tickDistance = _, lasso) : tickDistance; }\n  lasso.color = function(_) { return arguments.length ? (color = _, lasso) : color; }\n  lasso.animationRate = function(_) { return arguments.length ? (animationRate = _, lasso) : animationRate; }\n  lasso.opacity = function(_) { return arguments.length ? (opacity = _, lasso) : opacity; }\n  lasso.dashArray = function(_) { return arguments.length ? (dashArray = _, lasso) : dashArray; }\n  lasso.stroke = function(_) { return arguments.length ? (stroke = _, lasso) : stroke; }\n  lasso.lassoedFill = function(_) { return arguments.length ? (lassoedFill = _, lasso) : lassoedFill; }\n  lasso.lassoedStroke = function(_) { return arguments.length ? (lassoedStroke = _, lasso) : lassoedStroke; }\n  lasso.lassoedStrokeWidth = function(_) { return arguments.length ? (lassoedStrokeWidth = _, lasso) : lassoedStrokeWidth; }\n  lasso.eventCatcher = function(_) { return arguments.length ? (eventCatcher = _, lasso) : eventCatcher; }\n\n  lasso.drag = drag\n  lasso.draw = draw\n  lasso.tick = tick\n  lasso.detect = detect\n  lasso.toggle = toggle\n  lasso.remove = remove\n  lasso.render = render\n  lasso.keyFrames = keyFrames\n  lasso.updateObjects = updateObjects\n  lasso.applyPathAttributes = applyPathAttributes\n  lasso.applyObjectAttributes = applyObjectAttributes\n\n  keyFrames()\n\n  function lasso() {\n    // add a dash animation if needed\n    if (activeQ) { transitionDraw() }\n  }\n\n  function toggle(state) {\n    // use optional param to set state, otherwise toggle state\n    activeQ = (state!=undefined) ? state : !activeQ\n    chartOffset = getTranslation(chartContainer)\n    objectsOffset = getTranslation(objectContainer)\n\n    if (activeQ) {\n      svg.node().addEventListener('mousedown', render, true)\n    } else {\n      svg.node().removeEventListener('mousedown', render, true)\n      remove()\n    }\n\n  }\n\n  function draw() {\n    chartOffset = getTranslation(chartContainer)\n    objectsOffset = getTranslation(objectContainer)\n\n    var container = safeSelect(objectContainer, 'g', 'lasso-container')\n    var paths = container.selectAll('path[instance=\"'+instance+'\"]')\n\n    // update\n    paths = paths.data(allPoints)\n\n    // remove excess\n    var pExit = paths.exit().remove()\n    // add needed paths\n    var pEnter = paths.enter().append('path')\n\n    // merge\n    paths = paths.merge(pEnter)\n\n    // apply\n    applyPathAttributes(paths)\n  }\n\n  function remove() {\n    var container = safeSelect(objectContainer, 'g', 'lasso-container')\n    var paths = container.selectAll('path[instance=\"'+instance+'\"]').remove()\n    container.remove()\n    objectContainer.selectAll(objectClass).classed(\"in-lasso\", false)\n    updateObjects()\n  }\n\n  function render( event ) {\n    // nothing can interefer with drawing the lasso\n    event.preventDefault(); event.stopPropagation();\n\n    var container = safeSelect(objectContainer, 'g', 'lasso-container')\n\n    /*\n    each time the user presses down, while the state is active, the lasso\n    the lasso should make a seperate segment.\n    */\n    currentPoints = [];\n\n    svg.node().addEventListener('mousemove', drag)\n    svg.node().addEventListener('mouseup', function(event) {\n      svg.node().removeEventListener('mousemove', drag)\n      allPoints.push(currentPoints)\n      // BUG:  somehow this is pushing currentPoints n times where n is the nth lasso path for the current instance\n      // NOTE: allPoints = unique(allPoints) is a temporary and inefficient fix\n      allPoints = unique(allPoints)\n    })\n\n    path = container.append('path').data([currentPoints])\n    applyPathAttributes(path)\n  }\n\n  function transitionDraw() {\n    var container = safeSelect(objectContainer, 'g', 'lasso-container')\n    var paths = container.selectAll('path[instance=\"'+instance+'\"]')\n\n    // update\n    paths = paths.data(allPoints)\n\n    // remove excess\n    var pExit = paths.exit().remove()\n    // add needed paths\n    var pEnter = paths.enter().append('path')\n\n    // merge\n    paths = paths.merge(pEnter)\n    .transition().duration(transitionDuration)\n    .ease(easeFunc)\n    applyPathAttributes(paths)\n\n  }\n\n  function applyPathAttributes(path) {\n    path\n    .attr(\"class\", hypenate(namespace, \"lasso-path\"))\n    .style('opacity', opacity)\n    .attr('fill', color)\n    .attr(\"d\", line)\n    .attr('instance', instance)\n    .style(\"stroke-dasharray\", dashArray)\n    .attr(\"stroke\", stroke)\n    .attr(\"stroke-width\", strokeWidth)\n    .style('animation', 'lassoDash '+animationRate+' linear')\n    .style(\"animation-iteration-count\", \"infinite\")\n  }\n\n  function drag(event) {\n    /*\n    effectively create a mouse down and move event (which normally is inteperated\n    as 'drag' by the browser) by dynamically adding / removing this event on\n    mouse down / mouse up.\n    */\n\n    if (eventCatcher != undefined) {eventCatcher.dispatch(hypenate(namespace,\"drag\"))}\n    // d3.dispatch(hypenate(namespace,\"drag\"))\n\n    if (event.which != 1) {return} // ensures left mouse button set\n    d3.event = event\n    var pt = d3.mouse(objectContainer.node());\n    var pt = d3.mouse(svg.node());\n\n    if (xScale != undefined) {pt[0] = xScale.invert(pt[0])}\n    if (yScale != undefined) {pt[1] = yScale.invert(pt[1])}\n    pt[0] = pt[0] - chartOffset[0] - objectsOffset[0]\n    pt[1] = pt[1] - chartOffset[1] - objectsOffset[1]\n\n    /* if we have a point already, test if it passes a minimum distance to prevent overwhelming with too many tick functions */\n    if (currentPoints.length) {\n      var lastPt = currentPoints[currentPoints.length - 1]\n      var a = [pt[0], pt[1]], b = [lastPt[0], lastPt[1]]\n\n      if (xScale) {b[0] = xScale(b[0]); a[0] = xScale(a[0])}\n      if (yScale) {b[1] = yScale(b[1]); a[1] = yScale(a[1])}\n\n      var dist = euclideanDistance(b, a)\n      if (dist > tickDistance) { tick(pt) }\n    }\n    else { tick(pt) }\n  }\n\n\n  function tick (pt) {\n    /*\n    If a point is provided update data and objects.\n    Otherwise just call on data we already have.\n\n    Why like this?:\n    1. currentPoints is current points to allow disjunct lassos, currentPoints is only pushed to\n    allPoints after mouseup.\n    2. to allow render of objects in the lasso class / updating the data list\n    just by toggling the button\n    */\n\n    if (pt != undefined) {\n      currentPoints.push(pt);\n      path.attr(\"d\", line);\n      if (currentPoints.length < 3) {return} // need at least 3 points to detect anything.\n      detect(allPoints.concat([currentPoints]))\n    } else {\n      detect(allPoints)\n    }\n  }\n\n\n  function detect(lassos) {\n    if (lassos == undefined) {lassos = allPoints}\n    objectContainer.selectAll(objectClass).each(function(d, i){\n      var current = d3.select(this),\n\n      box = current.absolutePosition(),\n      // box = current.relativePositionTo(objectContainer.node()),\n\n      boxPts = [\n        [\n          box.left - chartOffset[0] - objectsOffset[0],\n          box.top - chartOffset[1] - objectsOffset[1]\n        ],\n        [\n          box.right - chartOffset[0] - objectsOffset[0],\n          box.top - chartOffset[1] - objectsOffset[1]\n        ],\n        [\n          box.left - chartOffset[0] - objectsOffset[0],\n          box.bottom - chartOffset[1] - objectsOffset[1]\n        ],\n        [\n          box.right - chartOffset[0] - objectsOffset[0],\n          box.bottom - chartOffset[1] - objectsOffset[1]\n        ]\n      ]\n\n      if (xScale != undefined) {\n        boxPts[0][0] = xScale.invert(boxPts[0][0])\n        boxPts[1][0] = xScale.invert(boxPts[1][0])\n        boxPts[2][0] = xScale.invert(boxPts[2][0])\n        boxPts[3][0] = xScale.invert(boxPts[3][0])\n      }\n      if (yScale != undefined) {\n        boxPts[0][1] = yScale.invert(boxPts[0][1])\n        boxPts[1][1] = yScale.invert(boxPts[1][1])\n        boxPts[2][1] = yScale.invert(boxPts[2][1])\n        boxPts[3][1] = yScale.invert(boxPts[3][1])\n      }\n\n\n      /*\n      flag needed as we have to test multiple lasso segments, and if the point\n      is not in one segment, it does not mean it is not in any\n      */\n      var inAnyLassoQ = false;\n      for (var i = 0; i < lassos.length; i++) {\n        var lassoPoints = lassos[i]\n        // .map(function(pt){\n        //   var x, y = pt\n        //   if (xScale!=undefined) {x = xScale(x)}\n        //   if (yScale!=undefined) {y = yScale(y)}\n        //   return [x, y]\n        // })\n        var boxInLassoQ = boxPts.every(coord => d3.polygonContains(lassoPoints, coord))\n\n        if (boxInLassoQ) { inAnyLassoQ = true; } // only update flag in the positive case.\n      }\n\n      current.classed('in-lasso', inAnyLassoQ)\n      current.classed('in-lasso-'+instance, inAnyLassoQ)\n    })\n\n    updateObjects()\n    return objectContainer.selectAll('.in-lasso-'+instance)\n  }\n\n\n\n  function updateObjects() {\n    objectContainer.selectAll(objectClass).each(function(d, i) {\n      var t = d3.select(this)\n      applyObjectAttributes(t, t.classed('in-lasso'))\n    })\n  }\n\n  function applyObjectAttributes(obj, setQ) {\n    var\n    preLassoFill = obj.attr('_pre_lasso_fill'),\n    preLassoStroke = obj.attr('_pre_lasso_stroke'),\n    preLassoStrokeWidth = obj.attr('_pre_lasso_stroke-width')\n\n    if (setQ) {\n      obj.classed(\"in-lasso\", true)\n      obj.classed('in-lasso-'+instance, true)\n      if (preLassoFill == undefined) { obj.attr('_pre_lasso_fill', obj.attr('fill')) }\n      if (preLassoStroke == undefined) { obj.attr('_pre_lasso_stroke', obj.attr('stroke')) }\n      if (preLassoStrokeWidth == undefined) { obj.attr('_pre_lasso_stroke-width', obj.attr('stroke-width')) }\n\n      obj\n      //BUG: when .raise()\n      .attr('fill', lassoedFill)\n      .attr('stroke', lassoedStroke)\n      .attr('stoke-width', lassoedStrokeWidth)\n\n    } else {\n      obj.classed(\"in-lasso\", false)\n      obj.classed('in-lasso-'+instance, false)\n      if (preLassoFill != undefined) { obj.attr('fill', preLassoFill) }\n      if (preLassoStroke != undefined) { obj.attr('stroke', preLassoStroke) }\n      if (preLassoStrokeWidth != undefined) { obj.attr('stroke-width', preLassoStrokeWidth) }\n    }\n  }\n\n  function keyFrames() {\n    var style =\n    d3.select(\"html\").select('style.'+hypenate(namespace,\"lasso-dash\"))\n    if (style.empty()) {\n      d3.select(\"html\").append('style')\n      .classed(hypenate(namespace,\"lasso-dash\"), true)\n      .html(\"@keyframes lassoDash {to { stroke-dashoffset: 1000;}}\")\n    }\n\n  }\n  return lasso\n}\n","import {getContainingSVG} from \"./helpers\";\n/*******************************************************************************\n**                                                                            **\n**                                                                            **\n**                             D3 EXTENSIONS                                  **\n**                                                                            **\n**                                                                            **\n*******************************************************************************/\n/**\n* Recursively ascends parents of selection until it finds an svg tag\n* @function d3.selection.thisSVG\n* @augments d3.selection\n* @returns {Element} which is the svg tag, not the d3 selection of that tag\n*/\nd3.selection.prototype.thisSVG = function() { return getContainingSVG(this.node()); }\n\n\n/**\n* Helper for getting absolute position of the mouse\n* @function d3.mouse.absolute\n* @augments d3.mouse\n* @returns {number[]} [x, y] as they relate to `html` not to local scope.\n*/\nd3.mouse.absolute = function() {\n  var html = d3.select('html').node()\n  var [x, y] = this(html)\n  return [x, y]\n}\n\n\n/**\n* Gets position of the selection in relation to the containing svg\n* @see{@link getContainingSVG}\n* @function d3.selection.absolutePosition\n* @augments d3.selection\n* @returns {Object} with structure similar to getBoundingClientRect, e.g.\n* top, left, bottom, right, height, width\n*/\nd3.selection.prototype.absolutePosition = function() {\n    var element = this.node();\n    var elementPosition = element.getBoundingClientRect();\n    var containerSVG = getContainingSVG(element)\n    var svgPosition = containerSVG.getBoundingClientRect();\n\n    return {\n        top:    elementPosition.top    - svgPosition.top,\n        left:   elementPosition.left   - svgPosition.left,\n        bottom: elementPosition.bottom - svgPosition.top,\n        right:  elementPosition.right  - svgPosition.left,\n        height: elementPosition.height,\n        width:  elementPosition.width\n    };\n\n}\n\n\nd3.selection.prototype.relativePositionTo = function(container) {\n    var element = this.node();\n    var elementPosition = element.getBoundingClientRect();\n    var containerSVG = container\n    var svgPosition = containerSVG.getBoundingClientRect();\n\n    return {\n        top:    elementPosition.top    - svgPosition.top,\n        left:   elementPosition.left   - svgPosition.left,\n        bottom: elementPosition.bottom - svgPosition.top,\n        right:  elementPosition.right  - svgPosition.left,\n        height: elementPosition.height,\n        width:  elementPosition.width\n    };\n\n}\n","// Import styles (automatically inject into <head>).\n// import '../styles/main.css';\nimport {axis} from './modules/axis';\nimport {bar} from './modules/bar';\nimport {bubbleHeatmap} from './modules/bubble-heatmap';\nimport {heatmap} from './modules/heatmap';\nimport {boxwhisker} from './modules/box-whisker';\nimport {colorFunction} from './modules/color-function';\nimport {datatoggle} from './modules/data-toggle';\nimport {groupingSpacer} from './modules/grouping-spacer';\nimport {tooltip} from './modules/tooltip';\nimport {scatter} from './modules/scatter';\nimport {plotZoom} from './modules/plot-zoom';\nimport {multiPlotZoom} from './modules/multi-plot-zoom';\nimport {violin} from './modules/violin';\nimport {numericLegend} from './modules/numeric-legend';\nimport {categoricLegend} from './modules/categorical-legend';\nimport {lasso} from './modules/lasso';\nimport {lassoWidget} from './modules/lasso-widget';\nimport {selectFilter} from './modules/select-filter';\nimport {upset} from './modules/upset';\nimport {filterTable} from './modules/filter-table';\n\nimport {uniqueElements, getTranslation, modifyHexidecimalColorLuminance, tickRange,\nquartiles, extractViolinValues, hypenate, round, getContainingSVG,\ninterpolateColors, truncateText, safeSelect} from './modules/helpers';\n\nimport {\n  all, tally, hasQ, first, last, total, unique, get, listOfListsQ,\n  cut, groupBy, arrayEquals, elementsAtLevels, numberOfElements,\n  flatten, whichBin\n} from './modules/array-functions';\n\n\nimport {\n  setupStandardChartContainers, log as myLog, warn, info, error,\n  consoleGroup, consoleGroupEnd, resizeDebounce\n} from './modules/utils';\n\n// /** @module d3sm */\nvar d3sm = {};\nd3sm.axis = axis;\nd3sm.bar = bar;\nd3sm.bubbleHeatmap = bubbleHeatmap;\nd3sm.heatmap = heatmap;\nd3sm.boxwhisker = boxwhisker;\nd3sm.colorFunction = colorFunction;\nd3sm.datatoggle = datatoggle;\nd3sm.groupingSpacer = groupingSpacer;\nd3sm.tooltip = tooltip;\nd3sm.scatter = scatter;\nd3sm.plotZoom = plotZoom;\nd3sm.multiPlotZoom = multiPlotZoom;\nd3sm.violin = violin;\nd3sm.numericLegend = numericLegend;\nd3sm.categoricLegend = categoricLegend;\nd3sm.lasso = lasso;\nd3sm.lassoWidget = lassoWidget;\nd3sm.selectFilter = selectFilter;\nd3sm.upset = upset;\nd3sm.filterTable = filterTable;\n\nd3sm.uniqueElements = uniqueElements;\nd3sm.getTranslation = getTranslation;\nd3sm.modifyHexidecimalColorLuminance = modifyHexidecimalColorLuminance;\nd3sm.tickRange = tickRange;\nd3sm.quartiles = quartiles;\nd3sm.extractViolinValues = extractViolinValues;\nd3sm.hypenate = hypenate;\nd3sm.round = round;\nd3sm.getContainingSVG = getContainingSVG;\nd3sm.interpolateColors = interpolateColors;\nd3sm.truncateText = truncateText;\nd3sm.safeSelect = safeSelect;\n\nd3sm.whichBin = whichBin;\nd3sm.unique = unique;\nd3sm.flatten = flatten;\n\nd3sm.setupStandardChartContainers = setupStandardChartContainers;\nd3sm.log = myLog;\nd3sm.warn = warn;\nd3sm.info = info;\nd3sm.error = error;\nd3sm.consoleGroup = consoleGroup;\nd3sm.consoleGroupEnd = consoleGroupEnd;\nd3sm.resizeDebounce = resizeDebounce;\n\nd3sm.debugQ = false\n\n\n\n// Import a logger for easier debugging\n// import debug from 'debug';\n// const log = debug('app:log');\n\n// The logger should only be disabled if we're not in production.\n// if (ENV !== 'production') {\n//   // Enable the logger.\n//   debug.enable('*');\n//   log('Logging is enabled!');\n//\n//   // Enable LiveReload\n//   document.write(\n//     '<script src=\"http://'\n//     + (location.host || 'localhost').split(':')[0]\n//     + ':35729/livereload.js?snipver=1\"></'\n//     + 'script>'\n//   );\n// } else {\n//   debug.disable();\n// }\n\nwindow.d3sm = d3sm;\n\nexport default d3sm\n","import {\n  hypenate, safeSelect, extractViolinValues,\n  tickRange, modifyHexidecimalColorLuminance, truncateText,\n  truncateString,\n  round\n} from './helpers';\nimport {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer} from './utils';\nimport {unique, hasQ, flatten} from './array-functions';\nimport {groupingSpacer} from './grouping-spacer';\n/*******************************************************************************\n**                                                                            **\n**                                                                            **\n**                                  AXIS                                      **\n**                                                                            **\n**                                                                            **\n*******************************************************************************/\n\n/**\n * Creates an axis\n *\n * {@link https://sumneuron.gitlab.io/d3sm/demos/axes/index.html Demo}\n * @constructor axis\n * @param {d3.selection} selection\n * @namespace axis\n * @returns {function} axis\n */\nexport function axis ( selection ) {\n  var\n  /**\n  * The orientation of the axis\n  * (see {@link axis#orient})\n  * @param {string} [orient='bottom']\n  * @memberof axis#\n  * @property\n  */\n  orient = 'bottom',       // direction of the axis\n\n  /**\n  * Amount of horizontal space (in pixels) avaible to render the axis in\n  * (see {@link axis#spaceX})\n  * @param {number} [spaceX=0]\n  * @memberof axis#\n  * @property\n  */\n  spaceX=0,\n  /**\n  * Amount of vertical space (in pixels) avaible to render the axis in\n  * (see {@link axis.spaceY})\n  * @param {number} [spaceY=0]\n  * @memberof axis#\n  * @property\n  */\n  spaceY=0,\n\n\n  /**\n  * Whether or not to allow axis to render elements pass the main spatial dimension\n  * given the orientation (see {@link axis#orient}), where {@link axis#orient}=\"bottom\" or {@link axis#orient}=\"top\"\n  * the main dimension is {@link axis#spaceX} and where {@link axis#orient}=\"left\" or {@link axis#orient}=\"right\"\n  * the main dimension is {@link axis#spaceY}\n  * @param {boolean} [overflowQ=false]\n  * @memberof axis#\n  * @property\n  */\n  overflowQ = false,    // whether or not to allow overflow\n  /**\n  * Whether or not the axis labels are for categorical data. If false,\n  * will use {@link axis#scale} to position ticks.\n  * @param {boolean} [categoricalQ=false]\n  * @memberof axis#\n  * @property\n  */\n  categoricalQ = false, // whether or not the axis is showing values or groups\n  /**\n  * Whether or not the axis ticks should have guidelines\n  * @param {boolean} [categoricalQ=false]\n  * @memberof axis#\n  * @property\n  */\n  guideLinesQ = false,    // whether or not to allow overflow\n\n\n  /**\n  * How to group the tick labels\n  * @param {Array[]} [grouping=undefined] list of putatively other lists, which should correspond to tickLabels\n  * will space tick labels in nested lists closer together than outer lists\n  * @memberof axis#\n  * @property\n  */\n  grouping,\n\n  /**\n  * The scale for which non-categorial (see {@link axis#categoricalQ}) ticks should be spaced\n  * @param {d3.scale} [scale=d3.scaleLinear]\n  * @memberof axis#\n  * @property\n  */\n\n  scale = d3.scaleLinear(),\n  /**\n  * The padding for the domain of the scale (see {@link axis#scale})\n  * @param {d3.scale} [scale=d3.scaleLinear]\n  * @memberof axis#\n  * @property\n  */\n  domainPadding = 0.5,\n\n\n  /**\n  * Default space for the spacer (percentage) of main dimension given the orientation\n  * (see {@link axis#orient}), where {@link axis#orient}=\"bottom\" or {@link axis#orient}=\"top\"\n  * the main dimension is {@link axis#spaceX} and where {@link axis#orient}=\"left\" or {@link axis#orient}=\"right\"\n  * the main dimension is {@link axis#spaceY}between ticks\n  * @param {number} [objectSpacer=0.05]\n  * @memberof axis#\n  * @property\n  */\n  objectSpacer = 0.05,\n  /**\n  * The minimum size that an object can be if {@link axis#categoricalQ} is set to true\n  * @param {number} [minObjectSize=15]\n  * @memberof axis#\n  * @property\n  */\n  minObjectSize = 15,\n  /**\n  * The maximum size that an object can be if {@link axis#categoricalQ} is set to true\n  * @param {number} [maxObjectSize=15]\n  * @memberof axis#\n  * @property\n  */\n  maxObjectSize = 50,\n\n  /**\n  * Color of the background\n  * @param {string} [backgroundFill=\"transparent\"]\n  * @memberof axis#\n  * @property\n  */\n  backgroundFill = 'transparent',\n  /**\n  * Namespace for all items made by this instance of axis\n  * @param {string} [namespace=\"d3sm-axis\"]\n  * @memberof axis#\n  * @property\n  */\n  namespace = 'd3sm-axis',\n  /**\n  * Class name for tick container (<g> element)\n  * @param {string} [objectClass=\"tick-group\"]\n  * @memberof axis#\n  * @property\n  */\n  objectClass = 'tick-group',\n\n  /**\n  * Values to show at each tick. Only used if categoricalQ is set true. See {@link axis#categoricalQ}\n  * @param {string[]} [tickLabels=undefined]\n  * @memberof axis#\n  * @property\n  */\n  tickLabels,   // what to place at ticks\n  /**\n  * Values to show at each tick. Only used if categoricalQ is set false. See {@link axis#categoricalQ}\n  * @param {string[] | number[]} [objectClass=undefined]\n  * @memberof axis#\n  * @property\n  */\n  tickValues,   // where to place ticks if not\n  /**\n  * Number of ticks to display if categoricalQ is false. See {@link axis#categoricalQ}\n  * @param {number} [numberOfTicks=5]\n  * @memberof axis#\n  * @property\n  */\n  numberOfTicks = 5,\n\n\n  /**\n  * Stroke color of the main axis line\n  * @param {string} [lineStroke='black']\n  * @memberof axis#\n  * @property\n  */\n  lineStroke = 'black',\n  /**\n  * Stroke width of the main axis line\n  * @param {number} [lineStrokeWidth=3]\n  * @memberof axis#\n  * @property\n  */\n  lineStrokeWidth = 3,\n\n\n  /**\n  * Stroke color of ticks\n  * @param {string} [tickStroke='black']\n  * @memberof axis#\n  * @property\n  */\n  tickStroke = 'black',\n  /**\n  * Stroke number of ticks\n  * @param {string} [tickStrokeWidth=2]\n  * @memberof axis#\n  * @property\n  */\n  tickStrokeWidth = 2,\n  /**\n  * Length - in pixels - of ticks\n  * @param {number} [tickLength=10]\n  * @memberof axis#\n  * @property\n  */\n  tickLength = 10,\n\n  tickTickLabelSpacer = 10,\n  tickLabelMargin = 10,\n\n\n  /**\n  * Font size of tick labels\n  * @param {number} [tickLabelFontSize=14]\n  * @memberof axis#\n  * @property\n  */\n  tickLabelFontSize = 14,\n  /**\n  * Min font size of tick labels\n  * @param {number} [tickLabelMinFontSize=8]\n  * @memberof axis#\n  * @property\n  */\n  tickLabelMinFontSize = 8,\n  /**\n  * Max font size of tick labels\n  * @param {number} [tickLabelMaxFontSize=20]\n  * @memberof axis#\n  * @property\n  */\n  tickLabelMaxFontSize = 20,\n\n\n  /**\n  * Text anchor of tick labels\n  * @param {string} [tickLabelTextAnchor=\"middle\"]\n  * @memberof axis#\n  * @property\n  */\n  tickLabelTextAnchor,\n  /**\n  * Rotation of tick labels\n  * @param {number} [tickLabelRotation=0]\n  * @memberof axis#\n  * @property\n  */\n  tickLabelRotation,\n  /**\n  * Optional function for extracting the tick label from data\n  * @param {function} [tickLabelFunc=undefined]\n  * @memberof axis#\n  * @property\n  */\n  tickLabelFunc = undefined,\n\n  /**\n  * Optional function for what to do when label is clicked\n  * @param {function} [tickLabelOnClick=function(d, i){}]\n  * @memberof axis#\n  * @property\n  */\n  tickLabelOnClick = function(d, i){},\n\n  /**\n  * Optional function for what to do when label is hovered\n  * @param {function} [tickLabelOnHoverFunc=function(d, i){}]\n  * @memberof axis#\n  * @property\n  */\n  tickLabelOnHoverFunc = function(d, i){\n    return String(d).replace('-', ' ').replace('_', ' ')\n  },\n\n\n  /**\n  * Length of guidelines\n  * @param {function} [guidelineSpace=undefined]\n  * @memberof axis#\n  * @property\n  */\n  guidelineSpace,\n  /**\n  * Stroke color of guidlines\n  * @param {string} [guidelineSpace=\"#333333\"]\n  * @memberof axis#\n  * @property\n  */\n  guideLineStroke = '#333333',\n  /**\n  * Stroke width of guidlines\n  * @param {number} [guidelineSpace=2]\n  * @memberof axis#\n  * @property\n  */\n  guideLineStrokeWidth = 2,\n\n  /**\n  * Duration of all transitions of this element\n  * @param {number} [transitionDuration=1000]\n  * @memberof axis#\n  * @property\n  */\n  transitionDuration = 1000,\n  /**\n  * Easing function for transitions\n  * @param {d3.ease} [easeFunc=d3.easeExp]\n  * @memberof axis#\n  * @property\n  */\n  easeFunc = d3.easeExp,\n\n\n  /**\n  * Closure variable for getting object size after calculation\n  * @param {number} [objectSize=undefined]\n  * @memberof axis#\n  * @property\n  */\n  objectSize,\n  /**\n  * Closure variable for getting spacer size after calculation\n  * @param {number} [spacerSize=undefined]\n  * @memberof axis#\n  * @property\n  */\n  spacerSize,\n\n  /**\n  * Decimal percision to round numerical tick labels to\n  * @param {number} [roundTo=2]\n  * @memberof axis#\n  * @property\n  */\n  roundTo = 2,\n\n  label,\n\n\n  reverseScaleQ = false\n\n  axis.label = function(_) { return arguments.length ? (label = _, axis) : label; };\n  axis.tickTickLabelSpacer = function(_) { return arguments.length ? (tickTickLabelSpacer = _, axis) : tickTickLabelSpacer; };\n  axis.tickLabelMargin = function(_) { return arguments.length ? (tickLabelMargin = _, axis) : tickLabelMargin; };\n  /**\n   * Gets or sets the selection in which items are manipulated\n   * @param {d3.selection} [_=none]\n   * @returns {axis | d3.selection}\n   * @memberof axis\n   * @property\n   * by default selection = selection\n   */\n\n  axis.selection = function(_) { return arguments.length ? (selection = _, axis) : selection; };\n\n  /**\n   * Gets or sets the orientation in which items are manipulated\n   * (see {@link axis#orient})\n   * @param {string} [_=none] should be horizontal or vertical\n   * @returns {axis | string}\n   * @memberof axis\n   * @property\n   * by default orient=\"bottom\"\n   */\n  axis.orient = function(_) { return arguments.length ? (orient = _, axis) : orient; };\n  /**\n   * Gets or sets the amount of horizontal space in which items are manipulated\n   * (see {@link axis#spaceX})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default spaceX = undefined\n   */\n  axis.spaceX = function(_) { return arguments.length ? (spaceX = _, axis) : spaceX; };\n  /**\n   * Gets or sets the amount of vertical space in which items are manipulated\n   * (see {@link axis#spaceY})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default spaceY = undefined\n   */\n  axis.spaceY = function(_) { return arguments.length ? (spaceY = _, axis) : spaceY; };\n\n\n  /**\n   * Gets / sets whether or not axis is allowed to go beyond specified dimensions\n   * (see {@link axis#spaceX})\n   * @param {boolean} [_=none]\n   * @returns {axis | boolean}\n   * @memberof axis\n   * @property\n   * by default overflowQ = false\n   */\n  axis.overflowQ = function(_) { return arguments.length ? (overflowQ = _, axis) : overflowQ; };\n  /**\n   * Gets / sets whether or not axis will display categorial ticks or by numerical value\n   * (see {@link axis#categoricalQ})\n   * @param {boolean} [_=none]\n   * @returns {axis | boolean}\n   * @memberof axis\n   * @property\n   * by default categoricalQ = false\n   */\n  axis.categoricalQ = function(_) { return arguments.length ? (categoricalQ = _, axis) : categoricalQ; };\n  /**\n   * Gets / sets whether or not axis ticks should have guidelines\n   * (see {@link axis#guideLinesQ})\n   * @param {boolean} [_=none]\n   * @returns {axis | boolean}\n   * @memberof axis\n   * @property\n   * by default guideLinesQ = false\n   */\n  axis.guideLinesQ = function(_) { return arguments.length ? (guideLinesQ = _, axis) : guideLinesQ; };\n\n\n  /**\n   * Gets / sets how ticks should be groupped\n   * (see {@link axis#grouping})\n   * @param {Array[]} [_=none] list of putatively other lists, which should correspond to tickLabels\n   * will space tick labels in nested lists closer together than outer lists\n   * @returns {axis | Array[]}\n   * @memberof axis\n   * @property\n   * by default grouping = undefined\n   */\n  axis.grouping = function(_) { return arguments.length ? (grouping = _, axis) : grouping; };\n\n\n  /**\n   * Gets / sets the scale for which non-categorial  ticks should\n   * be spaced\n   * (see {@link axis#scale})\n   * @param {d3.scale} [_=none]\n   * @returns {axis | d3.scale}\n   * @memberof axis\n   * @property\n   * by default scale = d3.scaleLinear()\n   */\n  axis.scale = function(_) { return arguments.length ? (scale = _, axis) : scale; };\n  /**\n   * Gets / sets the padding for the domain of the scale\n   * (see {@link axis#domainPadding})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default domainPadding = 0.5\n   */\n  axis.domainPadding = function(_) { return arguments.length ? (domainPadding = _, axis) : domainPadding; };\n\n\n  /**\n   * Gets / sets objectSpacer\n   * (see {@link axis#objectSpacer})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default objectSpacer = 0.05\n   */\n  axis.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, axis) : objectSpacer; };\n  /**\n   * Gets / sets the minObjectSize\n   * (see {@link axis#minObjectSize})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default minObjectSize = 15\n   */\n  axis.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, axis) : minObjectSize; };\n  /**\n   * Gets / sets the maxObjectSize\n   * (see {@link axis#maxObjectSize})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default maxObjectSize = 50\n   */\n  axis.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, axis) : maxObjectSize; };\n\n\n  /**\n   * Gets / sets the namespace\n   * (see {@link axis#namespace})\n   * @param {string} [_=none]\n   * @returns {axis | string}\n   * @memberof axis\n   * @property\n   * by default namespace = 'd3sm-axis'\n   */\n  axis.namespace = function(_) { return arguments.length ? (namespace = _, axis) : namespace; };\n  /**\n   * Gets / sets the backgroundFill\n   * (see {@link axis#backgroundFill})\n   * @param {string} [_=none]\n   * @returns {axis | string}\n   * @memberof axis\n   * @property\n   * by default backgroundFill = 'transparent'\n   */\n  axis.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, axis) : backgroundFill; };\n  /**\n   * Gets / sets the objectClass\n   * (see {@link axis#objectClass})\n   * @param {string} [_=none]\n   * @returns {axis | string}\n   * @memberof axis\n   * @property\n   * by default objectClass = 'tick-group'\n   */\n  axis.objectClass = function(_) { return arguments.length ? (objectClass = _, axis) : objectClass; };\n\n\n  /**\n   * Gets / sets the tickLabels\n   * (see {@link axis#tickLabels})\n   * @param {string[]} [_=none]\n   * @returns {axis | string[]}\n   * @memberof axis\n   * @property\n   * by default tickLabels = undefined\n   */\n  axis.tickLabels = function(_) { return arguments.length ? (tickLabels = _, axis) : tickLabels; };\n  /**\n   * Gets / sets the tickValues\n   * (see {@link axis#tickValues})\n   * @param {number[]} [_=none]\n   * @returns {axis | number[]}\n   * @memberof axis\n   * @property\n   * by default tickValues = undefined\n   */\n  axis.tickValues = function(_) { return arguments.length ? (tickValues = _, axis) : tickValues; };\n  /**\n   * Gets / sets the tickValues\n   * (see {@link axis#numberOfTicks})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default numberOfTicks = 5\n   */\n  axis.numberOfTicks = function(_) { return arguments.length ? (numberOfTicks = _, axis) : numberOfTicks; };\n\n\n  /**\n   * Gets / sets the lineStroke\n   * (see {@link axis#lineStroke})\n   * @param {string} [_=none]\n   * @returns {axis | string}\n   * @memberof axis\n   * @property\n   * by default lineStroke = 'black'\n   */\n  axis.lineStroke = function(_) { return arguments.length ? (lineStroke = _, axis) : lineStroke; };\n  /**\n   * Gets / sets the lineStrokeWidth\n   * (see {@link axis#lineStrokeWidth})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default lineStrokeWidth = 3\n   */\n  axis.lineStrokeWidth = function(_) { return arguments.length ? (lineStrokeWidth = _, axis) : lineStrokeWidth; };\n\n\n  /**\n   * Gets / sets the tickStroke\n   * (see {@link axis#tickStroke})\n   * @param {string} [_=none]\n   * @returns {axis | string}\n   * @memberof axis\n   * @property\n   * by default tickStroke = 'black'\n   */\n  axis.tickStroke = function(_) { return arguments.length ? (tickStroke = _, axis) : tickStroke; };\n  /**\n   * Gets / sets the tickStrokeWidth\n   * (see {@link axis#tickStrokeWidth})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default tickStrokeWidth = 2\n   */\n  axis.tickStrokeWidth = function(_) { return arguments.length ? (tickStrokeWidth = _, axis) : tickStrokeWidth; };\n  /**\n   * Gets / sets the tickLength\n   * (see {@link axis#tickLength})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default tickLength = 10\n   */\n  axis.tickLength = function(_) { return arguments.length ? (tickLength = _, axis) : tickLength; };\n\n\n  /**\n   * Gets / sets the tickLabelFontSize\n   * (see {@link axis#tickLabelFontSize})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default tickLabelFontSize = 14\n   */\n  axis.tickLabelFontSize = function(_) { return arguments.length ? (tickLabelFontSize = _, axis) : tickLabelFontSize; };\n  /**\n   * Gets / sets the tickLabelMinFontSize\n   * (see {@link axis#tickLabelMinFontSize})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default tickLabelMinFontSize = 8\n   */\n  axis.tickLabelMinFontSize = function(_) { return arguments.length ? (tickLabelMinFontSize = _, axis) : tickLabelMinFontSize; };\n  /**\n   * Gets / sets the tickLabelMaxFontSize\n   * (see {@link axis#tickLabelMaxFontSize})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default tickLabelMaxFontSize = 20\n   */\n  axis.tickLabelMaxFontSize = function(_) { return arguments.length ? (tickLabelMaxFontSize = _, axis) : tickLabelMaxFontSize;};\n\n\n  /**\n   * Gets / sets the tickLabelTextAnchor\n   * (see {@link axis#tickLabelTextAnchor})\n   * @param {string} [_=none]\n   * @returns {axis | string}\n   * @memberof axis\n   * @property\n   * by default tickLabelTextAnchor = 'center'\n   */\n  axis.tickLabelTextAnchor = function(_) { return arguments.length ? (tickLabelTextAnchor = _, axis) : tickLabelTextAnchor; };\n  /**\n   * Gets / sets the tickLabelRotation\n   * (see {@link axis#tickLabelRotation})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default tickLabelRotation = 0\n   */\n  axis.tickLabelRotation = function(_) { return arguments.length ? (tickLabelRotation = _, axis) : tickLabelRotation; };\n  /**\n   * Gets / sets the tickLabelFunc\n   * (see {@link axis#tickLabelFunc})\n   * @param {function} [_=none]\n   * @returns {axis | function}\n   * @memberof axis\n   * @property\n   * by default tickLabelFunc = undefined\n   */\n  axis.tickLabelFunc = function(_) { return arguments.length ? (tickLabelFunc = _, axis) : tickLabelFunc; };\n\n\n  /**\n  * Gets / sets the tickLabelOnClick\n  * (see {@link axis#tickLabelOnClick})\n  * @param {function} [_=none]\n  * @returns {axis | function}\n  * @memberof axis\n  * @property\n  */\n  axis.tickLabelOnClick = function(_) { return arguments.length ? (tickLabelOnClick = _, axis) : tickLabelOnClick; };\n\n\n  /**\n   * Gets / sets the guidelineSpace\n   * (see {@link axis#guidelineSpace})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default guidelineSpace = undefined\n   */\n  axis.guidelineSpace = function(_) { return arguments.length ? (guidelineSpace = _, axis) : guidelineSpace; };\n  /**\n   * Gets / sets the guideLineStroke\n   * (see {@link axis#guideLineStroke})\n   * @param {string} [_=none]\n   * @returns {axis | string}\n   * @memberof axis\n   * @property\n   * by default guideLineStroke = \"#333333\"\n   */\n  axis.guideLineStroke = function(_) { return arguments.length ? (guideLineStroke = _, axis) : guideLineStroke; };\n  /**\n   * Gets / sets the guideLineStrokeWidth\n   * (see {@link axis#guideLineStrokeWidth})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default guideLineStrokeWidth = 2\n   */\n  axis.guideLineStrokeWidth = function(_) { return arguments.length ? (guideLineStrokeWidth = _, axis) : guideLineStrokeWidth; };\n\n\n  /**\n   * Gets / sets the transitionDuration\n   * (see {@link axis#transitionDuration})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default transitionDuration = 1000\n   */\n  axis.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, axis) : transitionDuration; };\n  /**\n   * Gets / sets the easeFunc\n   * (see {@link axis#easeFunc})\n   * @param {d3.ease} [_=none]\n   * @returns {axis | d3.ease}\n   * @memberof axis\n   * @property\n   * by default easeFunc = d3.easeExp\n   */\n  axis.easeFunc = function(_) { return arguments.length ? (easeFunc = _, axis) : easeFunc; };\n\n\n  /**\n   * Gets / sets the objectSize\n   * (see {@link axis#objectSize})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default objectSize = undefined\n   */\n  axis.objectSize = function(_) { return arguments.length ? (objectSize = _, axis) : objectSize; };\n  /**\n   * Gets / sets the spacerSize\n   * (see {@link axis#spacerSize})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default spacerSize = undefined\n   */\n  axis.spacerSize = function(_) { return arguments.length ? (spacerSize = _, axis) : spacerSize; };\n\n  /**\n   * Gets / sets the roundTo\n   * (see {@link axis#roundTo})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default roundTo = 2\n   */\n   axis.roundTo = function(_) { return arguments.length ? (roundTo = _, axis) : roundTo; };\n   axis.reverseScaleQ = function(_) { return arguments.length ? (reverseScaleQ = _, axis) : reverseScaleQ; };\n\n\n   axis.tickLabelOnHoverFunc = function(_) {return arguments.length ? (tickLabelOnHoverFunc = _, axis) : tickLabelOnHoverFunc; };\n\n\n  function axis () {\n    // for convenience in handling orientation specific values\n    var horizontalQ = hasQ(['top', 'bottom', 'horizontal'], orient) ? true : false\n    var verticalQ = !horizontalQ\n\n    // background cliping rectangle\n    var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}\n    // modify the rect based on axis orientation\n    if (orient == \"left\") {\n      bgcpRect.x -= spaceX;\n      if(guideLinesQ) { bgcpRect.width += guidelineSpace };\n      /* these two lines increase the clipping rect to allow for text at the edge of the axis */\n      bgcpRect.y -= tickLabelMaxFontSize;\n      bgcpRect.height += 2*tickLabelMaxFontSize\n    }\n    if (orient == \"bottom\"){\n      bgcpRect.y = bgcpRect.y;\n      if(guideLinesQ) { bgcpRect.y -= guidelineSpace; bgcpRect.height += guidelineSpace; };\n      /* these two lines increase the clipping rect to allow for text at the edge of the axis */\n      bgcpRect.x -= tickLabelMaxFontSize;\n      bgcpRect.width += 2*tickLabelMaxFontSize\n    }\n    if (orient == \"top\") {\n      bgcpRect.y -= spaceY;\n      if(guideLinesQ) { bgcpRect.height += guidelineSpace };\n      /* these two lines increase the clipping rect to allow for text at the edge of the axis */\n      bgcpRect.y -= tickLabelMaxFontSize;\n      bgcpRect.height += 2*tickLabelMaxFontSize\n    }\n    if (orient == \"right\") { bgcpRect.x = 0;\n      if(guideLinesQ) { bgcpRect.width += guidelineSpace; bgcpRect.x -= guidelineSpace };\n      /* these two lines increase the clipping rect to allow for text at the edge of the axis */\n      bgcpRect.y -= tickLabelMaxFontSize;\n      bgcpRect.height += 2*tickLabelMaxFontSize\n    }\n\n\n    var container = setupContainer( selection, namespace, bgcpRect, backgroundFill );\n\n    // defaults for text-anchor and text rotation\n    if (orient == 'top') {\n      tickLabelTextAnchor = tickLabelTextAnchor == undefined ? 'start' : tickLabelTextAnchor\n      tickLabelRotation = tickLabelRotation == undefined ? -90 : tickLabelRotation\n    }\n    if (orient == 'bottom') {\n      tickLabelTextAnchor = tickLabelTextAnchor == undefined ? 'end' : tickLabelTextAnchor\n      tickLabelRotation = tickLabelRotation == undefined ? -90 : tickLabelRotation\n    }\n    if (orient == 'left') {\n      tickLabelTextAnchor = tickLabelTextAnchor == undefined ? 'end' : tickLabelTextAnchor\n      tickLabelRotation = tickLabelRotation == undefined ? 0 : tickLabelRotation\n    }\n    if (orient == 'right') {\n      tickLabelTextAnchor = tickLabelTextAnchor == undefined ? 'start' : tickLabelTextAnchor\n      tickLabelRotation = tickLabelRotation == undefined ? 0 : tickLabelRotation\n    }\n\n    /*\n    If categorical:\n      -> use grouping if defined,\n      -> else use the labels provided\n    else:\n      if grouping undefined\n        and no specified number of tickes\n          -> make numberOfTick ticks\n          -> else use provided tick values\n        -> use grouping\n    */\n    var tickData = categoricalQ\n    ? (grouping == undefined)\n      ? tickLabels\n      : grouping\n    : (grouping == undefined)\n      ? (numberOfTicks != undefined)\n      // ? (tickValues.length < numberOfTicks)\n        ? (tickRange(...d3.extent(tickValues), numberOfTicks))\n        : tickValues\n      : grouping\n\n\n    var flatTickData = flatten(tickData)\n    var numberOfObjects = flatTickData.length\n    var space = horizontalQ ? spaceX : spaceY\n    var extent = d3.extent(flatTickData)\n\n\n    if (reverseScaleQ) {extent.reverse()}\n    var domain = reverseScaleQ\n    ? [extent[0] + domainPadding, extent[1] - domainPadding]\n    : [extent[0] - domainPadding, extent[1] + domainPadding]\n\n    scale\n    .domain(domain)\n    .range([horizontalQ ? 0 : spaceY, horizontalQ ? spaceX : 0])\n\n\n    /*\n    Scales are based on the values of the chart and correspond to the spacings of the\n    chart. If the chart has already been rendered, these values (expensive to caluclate) can\n    be passed to axis to prevent recalculation.\n    */\n\n    // calculate object size if needed\n    objectSize = (objectSize == undefined)\n    ? calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ)\n    : objectSize\n\n    // calculate spacer size if needed\n    spacerSize = (spacerSize == undefined)\n    ? calculateWidthOfSpacer(flatTickData, space, objectSize, numberOfObjects, objectSpacer, overflowQ)\n    : spacerSize\n\n\n    var objClass = hypenate(namespace, categoricalQ ? objectClass+'-categorical' : objectClass)\n\n    var spacerFunction = groupingSpacer()\n    .horizontalQ(horizontalQ).scale(scale).moveby((categoricalQ?'category':'scale')).numberOfObjects(numberOfObjects)\n    .objectClass(objClass).objectSize(objectSize).spacerSize(spacerSize)\n    .transitionDuration(transitionDuration).easeFunc(easeFunc)\n    .namespace(namespace)\n\n    var tickEnterAnimation = function(sel){\n      var mt = scale(sel.datum()),\n      dist = scale(extent[1]) * 2,\n      k = (mt < extent[1] / 2) ? 1 : -1\n      k = horizontalQ ? k * -1 : k\n      sel.attr('transform', function (d, i) {\n        var\n        x = horizontalQ ?  dist * k : 0,\n        y = !horizontalQ ? dist * k : 0,\n        t = 'translate('+x+','+y+')'\n        return t\n      })\n    }\n    var tickExitAnimation = function(sel) {\n      var mt = scale(sel.datum()),\n      dist = scale(extent[1]) * 2,\n      k = (mt < extent[1] / 2) ? 1 : -1\n      k = horizontalQ ? k * -1 : k\n      sel.transition().duration(transitionDuration).ease(easeFunc)\n      .style('opacity', 0)\n      .attr('transform', function (d, i) {\n        var\n\n        x = horizontalQ ?  dist * k  : 0,\n        y = !horizontalQ ? dist * k : 0,\n        t = 'translate('+x+','+y+')'\n        return t\n      }).remove()\n    }\n\n    if (!categoricalQ){\n      spacerFunction.enterFunction(tickEnterAnimation)\n      spacerFunction.exitFunction(tickExitAnimation)\n    }\n\n    // move tick containers\n    spacerFunction(container, tickData, 0)\n\n    // move by for x and y needed to center categorical ticks, labels, and guidelines\n    function moveXBy(d, i, horizontalQ, categoricalQ, objectSize){\n      return (horizontalQ)\n      ? (categoricalQ)\n        ? objectSize / 2\n        : 0\n      : 0\n    }\n\n    function moveYBy(d, i, verticalQ, categoricalQ, objectSize){\n      return (verticalQ)\n      ? (categoricalQ)\n        ? objectSize / 2\n        : 0\n      : 0\n    }\n\n\n\n    var labelNameGroup = safeSelect(selection, 'g', hypenate(namespace,'axis-name'))\n\n\n\n    var labelElement = safeSelect(labelNameGroup, 'text', hypenate(namespace, 'name'))\n    if (labelElement != undefined) {\n      labelElement.text(label)\n\n      if (orient == 'left' || orient == 'right') {\n        labelElement.attr('transform', 'rotate(-90)')\n      }\n\n      var bbox = labelElement.node().getBoundingClientRect()\n      labelNameGroup.attr('transform',function(d, i){\n        var\n        x = 0,\n        y = 0,\n        t\n\n        if (orient == 'bottom') {\n          x = spaceX - bbox.width - tickLabelMargin\n          y = - tickTickLabelSpacer\n        }\n        else if (orient == 'top') {\n          x = spaceX - bbox.width - tickLabelMargin\n          x = tickLabelMargin\n          y = bbox.height + tickTickLabelSpacer\n        }\n        else if (orient == 'left') {\n          x = bbox.width + tickTickLabelSpacer\n          y = bbox.height + tickLabelMargin\n        } else if (orient == 'right') {\n          x = -(bbox.width + tickTickLabelSpacer)\n          y = bbox.height + tickLabelMargin\n        } else {\n\n        }\n        t = 'translate('+x+','+y+')'\n        return t\n      })\n\n\n\n    } else {\n      labelElement.remove()\n    }\n    /*\n    Idea from Stack Overflow\n    https://stackoverflow.com/questions/50579535/d3-js-v4-truncate-text-to-fit-in-fixed-space/50585022?noredirect=1#comment88235562_50585022\n    to use clip path to make things fit in fixed size. Have yet got this to work nicely.\n    */\n    // var defs = d3.select(container.node().parentNode).select('defs')\n    // var tickLabelClipPath = safeSelect(defs, 'clipPath', hypenate(namespace,'tick-label-clip-path')).attr('id',  hypenate(namespace,'tick-label-clip-path'))\n    // var tickLabelClipPathRect = safeSelect(tickLabelClipPath, 'rect',  hypenate(namespace,'tick-label-clip-path-rect'))\n    // .attr('x', 0)\n    // .attr('y', 0)\n    // .attr('width', function(d, i){\n    //   if (horizontalQ) { return tickLabelFontSize }\n    //   if (verticalQ) { return spaceX - tickLength }\n    // })\n    // .attr('height', function(d, i){\n    //   if (verticalQ) { return tickLabelFontSize }\n    //   if (horizontalQ) { return spaceY - tickLength }\n    // })\n\n\n    // for each tick container\n    var ticks = container.selectAll('g:not(.to-remove).'+objClass).each(function(d, i){\n      var that = d3.select(this).style('opacity', 1)\n\n      // make and move tick\n      var tick = safeSelect(that, 'line', hypenate(namespace,'tick'))\n      .attr(\"x1\", 0)\n      .attr(\"x2\", horizontalQ ? 0 : orient == \"left\" ? -tickLength : tickLength)\n      .attr(\"y1\", 0)\n      .attr('y2',  verticalQ ? 0 : orient == \"top\" ? -tickLength : tickLength)\n      .attr('stroke', tickStroke)\n      .attr('stroke-width', tickStrokeWidth)\n      .attr('transform', function(d, i) {\n        var\n        x = moveXBy(d, i, horizontalQ, categoricalQ, objectSize),\n        y = moveYBy(d, i, verticalQ, categoricalQ, objectSize),\n        t = 'translate('+x+','+y+')'\n        return t\n      })\n\n      // make and move label\n      var label = safeSelect(that, 'text', hypenate(namespace,'label'))\n      .text(function(d, i){\n        var s = typeof d == 'number' ? round(d, roundTo) : d\n        s = truncateString(String(s), (horizontalQ ? spaceY : spaceX) - tickLength-tickLabelMargin-tickTickLabelSpacer, tickLabelFontSize * 0.45)\n        return s\n      })\n      .attr('font-size', tickLabelFontSize)\n      .attr('text-anchor', tickLabelTextAnchor)\n      // truncateText(label, label.text(), orient, tickLength, horizontalQ ? spaceY : spaceX, overflowQ)\n\n      label.attr('transform', function(d, i) {\n        var\n        rect = d3.select(this).node().getBoundingClientRect(),\n        leng = d3.select(this).node().getComputedTextLength(),\n        x = moveXBy(d, i, horizontalQ, categoricalQ, objectSize),\n        y = moveYBy(d, i, verticalQ, categoricalQ, objectSize)\n        // on recall, rect changes because of rotation so need Math.min(rect.height, rect.width)\n\n        var s = Math.sin(tickLabelRotation) * leng * 0\n\n        if (orient == 'top') {\n          y = -(tickLength+tickTickLabelSpacer);\n          // y = tickLength+tickTickLabelSpacer;\n\n          // y -= Math.max(rect.height, rect.width);\n          x += Math.min(rect.height, rect.width) * 0.25\n          // x -= leng * 0.25 + s\n        }\n        if (orient == 'bottom') {\n          y = tickLength+tickTickLabelSpacer;\n          x += Math.min(rect.height, rect.width) * 0.25\n          // x += leng * 0.25 - s\n        }\n        if (orient == 'left') {\n          x -= (tickLength+tickTickLabelSpacer);\n          // y += rect.height * 0.5; y-= rect.height/4\n          y += Math.min(rect.height, rect.width) * 0.25\n          // y += leng * 0.25\n        }\n        if (orient == 'right') {\n          x += (tickLength+tickTickLabelSpacer);\n          // y += Math.min(rect.height, rect.width) * 0.25\n          // y += leng * 0.25\n          y += rect.height * 0.5; y-= rect.height/4\n        }\n\n        var\n        t = 'translate('+x+','+y+')',\n        r = 'rotate('+tickLabelRotation+')'\n        return t + r\n      })\n      .on('mousemove', labelHover)\n      .on('mouseout', labelHoverOff)\n      .on('click', tickLabelOnClick)\n      // .attr('clip-path', 'url(#'+hypenate(namespace,'tick-label-clip-path')+')')\n\n      // add guidlines as needed\n       if (guideLinesQ) {\n         var gline = safeSelect(that, 'line', hypenate(namespace, 'guideline'))\n         .transition().duration(transitionDuration).ease(easeFunc)\n         .attr(\"x1\", 0)\n         .attr(\"x2\", horizontalQ ? 0 : orient == \"left\" ? guidelineSpace : -guidelineSpace)\n         .attr(\"y1\", 0)\n         .attr('y2',  verticalQ ? 0 : orient == \"top\" ? guidelineSpace : -guidelineSpace)\n         .attr('transform', function(d, i) {\n           var\n           x = moveXBy(d, i, horizontalQ, categoricalQ, objectSize),\n           y = moveYBy(d, i, verticalQ, categoricalQ, objectSize),\n           t = 'translate('+x+','+y+')'\n           return t\n         })\n       } else { that.select('line.'+hypenate(namespace, 'guideline')).remove() }\n\n    })\n\n    // apply alternating guidline thickness\n    if (guideLinesQ) {\n      container.selectAll('.'+hypenate(namespace,'guideline'))\n      .attr('stroke', function(d, i){\n        if (i % 2 == 0) { return modifyHexidecimalColorLuminance(guideLineStroke, 0.8) }\n        return guideLineStroke\n      })\n      .attr('stroke-width', function(d, i){\n        if (i % 2 == 0) { return guideLineStrokeWidth *0.8}\n        return guideLineStrokeWidth\n      })\n      .attr('minor', function(d, i){return i%2 == 0})\n    }\n\n\n    /***************************************************************************\n    ** Make the line of the axis\n    ***************************************************************************/\n    var line = safeSelect(selection, 'path', hypenate(namespace,'line'))\n    // .attr('x1', 0)\n    // .attr('x2', horizontalQ ? spaceX : 0)\n    // .attr('y1', 0)\n    // .attr('y2', horizontalQ ? 0 : spaceY)\n    .attr('d',\n      horizontalQ\n      ? 'M 0,0 H' + spaceX + ',0'\n      : 'M 0,0 V 0,' + spaceY\n    )\n    .attr('stroke', lineStroke)\n    .attr('stroke-width', lineStrokeWidth)\n    .classed('axis-line', true)\n\n\n  }\n\n  // hover of label show full text label in case it is truncated\n  function labelHover(d, i){\n    var t = d3.select(this).style('fill', 'red')\n    d3.select(t.node().parentNode).select(\"line.\"+hypenate(namespace,'tick'))\n    .attr(\"stroke\", 'red')\n    .attr(\"stroke-width\", tickStrokeWidth*2)\n\n    if (guideLinesQ) {\n      d3.select(t.node().parentNode).select('line.'+hypenate(namespace, 'guideline'))\n      .attr('stroke', 'red')\n      .attr('stroke-width', guideLineStrokeWidth*2)\n    }\n\n    var s = typeof d == 'number' ? round(d, roundTo) : d\n\n    var m = d3.mouse(d3.select('html').node())\n    var div = safeSelect(d3.select('body'), 'div', hypenate(namespace,'guideline-tooltip'))\n    .attr('id', hypenate(namespace,'guideline-tooltip'))\n    .style('position', 'absolute')\n    .style('left', (d3.event.pageX+15)+'px')\n    .style('top', (d3.event.pageY+15)+'px')\n    .style('background-color', 'white')\n    .style('border-color', 'black')\n    // .style('min-width', (tickLabelFontSize * (String(s).split('.')[0].length+3))+'px')\n    // .style('min-height', (tickLabelFontSize * (String(s).split('.')[0].length+3))+'px')\n    .style('border-radius', '10px')\n    .style('display', 'flex')\n    .style('justify-content', 'center')\n    .style('text-align', 'middle')\n    .style('padding', 4+\"px\")\n\n    .style('border-style', 'solid')\n    .style('border-width', 2)\n\n    var text = safeSelect(div, 'div')\n    .text(tickLabelOnHoverFunc(s, i))\n    .style('color', 'black')\n    .style('align-self', 'center')\n\n    var bbox = div.node().getBoundingClientRect()\n    if (bbox.x + bbox.width > window.innerWidth) {\n      div.style('left', (d3.event.pageX-15-300)+'px')\n    }\n  }\n\n  function labelHoverOff(d, i){\n    var t = d3.select(this).style('fill', 'black')\n    d3.select(t.node().parentNode).select(\"line.\"+hypenate(namespace,'tick'))\n    .attr(\"stroke\", tickStroke)\n    .attr(\"stroke-width\", tickStrokeWidth)\n\n    if (guideLinesQ) {\n      var gline = d3.select(t.node().parentNode).select('line.'+hypenate(namespace, 'guideline'))\n      var minorQ = gline.attr('minor')\n      gline.attr('stroke', function(d, ii){\n        if (minorQ == 'true') { return modifyHexidecimalColorLuminance(guideLineStroke, 0.8) }\n        return guideLineStroke\n      })\n      .attr('stroke-width', function(d, ii){\n        if (minorQ == 'true') { return guideLineStrokeWidth *0.8}\n        return guideLineStrokeWidth\n      })\n    }\n    d3.select(\"#\"+hypenate(namespace,'guideline-tooltip')).remove()\n  }\n\n\n\n  return axis\n}\n","import {hypenate, safeSelect} from './helpers';\nimport {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer} from './utils';\nimport {unique, flatten} from './array-functions';\nimport {groupingSpacer} from './grouping-spacer';\nimport {colorFunction as CF} from './color-function';\nimport {tooltip as TTip} from './tooltip';\n/*******************************************************************************\n**                                                                            **\n**                                                                            **\n**                                   BAR                                      **\n**                                                                            **\n**                                                                            **\n*******************************************************************************/\n\n/**\n * Creates a bar\n *\n * {@link https://sumneuron.gitlab.io/d3sm/demos/bar-chart-same-data-complex-grouping/index.html Demo}\n * @constructor bar\n * @param {d3.selection} selection\n * @namespace bar\n * @returns {function} bar\n */\nexport function bar ( selection ) {\n  /*\n  Assumes that data is list an object.\n\n  The keys of data will be bound to the bars.\n  The valueExtractor function will extract the value from data[key].\n\n  Grouping can be used if desired. It should be an arbitrary complex list where\n  the values are strings matching keys in data.\n  */\n  var\n  /**\n  * Data to plot. Assumed to be a object, where each key corresponds to a bar\n  * (see {@link bar#data})\n  * @param {Object} [data=undefined]\n  * @memberof bar#\n  * @property\n  */\n  data,\n  /**\n  * Which direction to render the bars in\n  * (see {@link bar#orient})\n  * @param {number} [orient='horizontal']\n  * @memberof bar#\n  * @property\n  */\n  orient='horizontal',\n  /**\n  * Amount of horizontal space (in pixels) avaible to render the bar in\n  * (see {@link bar#spaceX})\n  * @param {number} [spaceX=undefined]\n  * @memberof bar#\n  * @property\n  */\n  spaceX,\n  /**\n  * Amount of vertical space (in pixels) avaible to render the bar in\n  * (see {@link bar.spaceY})\n  * @param {number} [spaceY=undefined]\n  * @memberof bar#\n  * @property\n  */\n  spaceY,\n\n  /**\n  * Whether or not to allow bar to render elements pass the main spatial dimension\n  * given the orientation (see {@link bar#orient}), where {@link bar#orient}=\"horizontal\"\n  * the main dimension is {@link bar#spaceX} and where {@link bar#orient}=\"vertical\"\n  * the main dimension is {@link bar#spaceY}\n  * @param {boolean} [overflowQ=false]\n  * @memberof bar#\n  * @property\n  */\n  overflowQ = false,\n\n  /**\n  * An array - putatively of other arrays - depicting how bars should be arranged\n  * @param {Array[]} [grouping=undefined]\n  * @memberof bar#\n  * @property\n  */\n  grouping,\n\n  /**\n  * How to get the value of the bar\n  * @param {function} [valueExtractor=function(key, index) { return data[key] }]\n  * @memberof bar#\n  * @property\n  */\n  valueExtractor = function(key, index) { return data[key] },\n  /**\n  * How to sort the bars - if {@link bar#grouping} is not provided.\n  * @param {function} [sortingFunction=function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}]\n  * @memberof bar#\n  * @property\n  */\n  sortingFunction = function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])},\n\n  /**\n  * The scale for which bar values should be transformed by\n  * @param {d3.scale} [scale=d3.scaleLinear]\n  * @memberof bar#\n  * @property\n  */\n  scale = d3.scaleLinear(),\n  /**\n  * The padding for the domain of the scale (see {@link bar#scale})\n  * @param {number} [domainPadding=0.5]\n  * @memberof bar#\n  * @property\n  */\n  domainPadding = 0.5,\n\n  /**\n  * Default space for the spacer (percentage) of main dimension given the orientation\n  * (see {@link bar#orient}), where {@link bar#orient}=\"horizontal\"\n  * the main dimension is {@link bar#spaceX} and where {@link bar#orient}=\"vertical\"\n  * the main dimension is {@link bar#spaceY} between bars\n  * @param {number} [objectSpacer=0.05]\n  * @memberof bar#\n  * @property\n  */\n  objectSpacer = 0.05,\n  /**\n  * The minimum size that an object can be\n  * @param {number} [minObjectSize=50]\n  * @memberof bar#\n  * @property\n  */\n  minObjectSize = 50,\n  /**\n  * The maximum size that an object can be\n  * @param {number} [maxObjectSize=100]\n  * @memberof bar#\n  * @property\n  */\n  maxObjectSize = 100,\n\n  /**\n  * The stroke width of the bars\n  * @param {number} [barStrokeWidth=2]\n  * @memberof bar#\n  * @property\n  */\n  barStrokeWidth = 2,\n  /**\n  * Instance of ColorFunction\n  * @param {function} [colorFunction = colorFunction()]\n  * @memberof bar#\n  * @property\n  */\n  colorFunction = CF(),\n\n\n  /**\n  * Color of the background\n  * @param {string} [backgroundFill=\"transparent\"]\n  * @memberof bar#\n  * @property\n  */\n  backgroundFill = 'transparent',\n  /**\n  * Namespace for all items made by this instance of bar\n  * @param {string} [namespace=\"d3sm-bar\"]\n  * @memberof bar#\n  * @property\n  */\n  namespace = 'd3sm-bar',\n  /**\n  * Class name for bar container (<g> element)\n  * @param {string} [objectClass=\"bar\"]\n  * @memberof bar#\n  * @property\n  */\n  objectClass = 'bar',\n\n  /**\n  * Duration of all transitions of this element\n  * @param {number} [transitionDuration=1000]\n  * @memberof bar#\n  * @property\n  */\n  transitionDuration = 1000,\n  /**\n  * Easing function for transitions\n  * @param {d3.ease} [easeFunc=d3.easeExp]\n  * @memberof bar#\n  * @property\n  */\n  easeFunc = d3.easeExp,\n\n  // useful values to extract to prevent re-calculation\n  /**\n  * The keys of the bars\n  * @param {string[]} [barKeys=undefined]\n  * @memberof bar#\n  * @property\n  */\n  barKeys,\n  /**\n  * The values of the bars\n  * @param {number[]} [barValues=undefined]\n  * @memberof bar#\n  * @property\n  */\n  barValues,\n  /**\n  * The objectSize (actual width) used by the bars\n  * @param {number} [objectSize=undefined]\n  * @memberof bar#\n  * @property\n  */\n  objectSize,\n  /**\n  * The spacerSize (actual width) used by the spacers between the bars\n  * @param {number} [spacerSize=undefined]\n  * @memberof bar#\n  * @property\n  */\n  spacerSize,\n  /**\n  * Instance of Tooltip\n  * @param {function} [tooltip=tooltip()]\n  * @memberof bar#\n  * @property\n  */\n  tooltip = TTip(),\n  barPercent = 1\n\n  /**\n   * Gets or sets the selection in which items are manipulated\n   * @param {d3.selection} [_=none]\n   * @returns {bar | d3.selection}\n   * @memberof bar\n   * @property\n   * by default selection = selection\n   */\n  bar.selection = function(_) { return arguments.length ? (selection = _, bar) : selection; };\n  /**\n   * Gets or sets the data\n   * (see {@link bar#data})\n   * @param {number} [_=none]\n   * @returns {bar | object}\n   * @memberof bar\n   * @property\n   */\n  bar.data = function(_) { return arguments.length ? (data = _, bar) : data; };\n  /**\n   * Gets or sets the orient of the bars\n   * (see {@link bar#orient})\n   * @param {number} [_=none]\n   * @returns {bar | object}\n   * @memberof bar\n   * @property\n   */\n  bar.orient = function(_) { return arguments.length ? (orient = _, bar) : orient; };\n  /**\n   * Gets or sets the amount of horizontal space in which items are manipulated\n   * (see {@link bar#spaceX})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {bar | number}\n   * @memberof bar\n   * @property\n   * by default spaceX = undefined\n   */\n  bar.spaceX = function(_) { return arguments.length ? (spaceX = _, bar) : spaceX; };\n  /**\n   * Gets or sets the amount of vertical space in which items are manipulated\n   * (see {@link bar#spaceY})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {bar | number}\n   * @memberof bar\n   * @property\n   * by default spaceY = undefined\n   */\n  bar.spaceY = function(_) { return arguments.length ? (spaceY = _, bar) : spaceY; };\n\n  /**\n   * Gets / sets whether or not bar is allowed to go beyond specified dimensions\n   * (see {@link bar#spaceX})\n   * @param {boolean} [_=none]\n   * @returns {bar | boolean}\n   * @memberof bar\n   * @property\n   * by default overflowQ = false\n   */\n  bar.overflowQ = function(_) { return arguments.length ? (overflowQ = _, bar) : overflowQ; };\n  /**\n   * Gets / sets the grouping of the bars\n   * (see {@link bar#grouping})\n   * @param {Array[]} [_=none]\n   * @returns {bar | Array[]}\n   * @memberof bar\n   * @property\n   * by default grouping = undefined\n   */\n  bar.grouping = function(_) { return arguments.length ? (grouping = _, bar) : grouping; };\n  /**\n   * Gets / sets the valueExtractor\n   * (see {@link bar#valueExtractor})\n   * @param {function} [_=none]\n   * @returns {bar | function}\n   * @memberof bar\n   * @property\n   * by default valueExtractor = function(key, index) { return data[key] },\n   */\n  bar.valueExtractor = function(_) { return arguments.length ? (valueExtractor = _, bar) : valueExtractor; };\n  /**\n   * Gets / sets the sortingFunction\n   * (see {@link bar#sortingFunction})\n   * @param {function} [_=none]\n   * @returns {bar | function}\n   * @memberof bar\n   * @property\n   * by default sortingFunction = function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])},\n   */\n  bar.sortingFunction = function(_) { return arguments.length ? (sortingFunction = _, bar) : sortingFunction; };\n  /**\n   * Gets / sets the scale for which the bar values should be transformed by\n   * (see {@link bar#scale})\n   * @param {d3.scale} [_=none]\n   * @returns {bar | d3.scale}\n   * @memberof bar\n   * @property\n   * by default scale = d3.scaleLinear()\n   */\n  bar.scale = function(_) { return arguments.length ? (scale = _, bar) : scale; };\n  /**\n   * Gets / sets the padding for the domain of the scale\n   * (see {@link bar#domainPadding})\n   * @param {number} [_=none]\n   * @returns {bar | number}\n   * @memberof bar\n   * @property\n   * by default domainPadding = 0.5\n   */\n  bar.domainPadding = function(_) { return arguments.length ? (domainPadding = _, bar) : domainPadding; };\n  /**\n   * Gets / sets objectSpacer\n   * (see {@link bar#objectSpacer})\n   * @param {number} [_=none]\n   * @returns {bar | number}\n   * @memberof bar\n   * @property\n   * by default objectSpacer = 0.05\n   */\n  bar.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, bar) : objectSpacer; };\n  /**\n   * Gets / sets the minObjectSize\n   * (see {@link bar#minObjectSize})\n   * @param {number} [_=none]\n   * @returns {bar | number}\n   * @memberof bar\n   * @property\n   * by default minObjectSize = 50\n   */\n  bar.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, bar) : minObjectSize; };\n  /**\n   * Gets / sets the maxObjectSize\n   * (see {@link bar#maxObjectSize})\n   * @param {number} [_=none]\n   * @returns {bar | number}\n   * @memberof bar\n   * @property\n   * by default maxObjectSize = 100\n   */\n  bar.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, bar) : maxObjectSize; };\n\n  /**\n   * Gets / sets the barStrokeWidth\n   * (see {@link bar#barStrokeWidth})\n   * @param {number} [_=none]\n   * @returns {bar | number}\n   * @memberof bar\n   * @property\n   * by default barStrokeWidth = 2\n   */\n  bar.barStrokeWidth = function(_) { return arguments.length ? (barStrokeWidth = _, bar) : barStrokeWidth; };\n  /**\n   * Gets / sets the colorFunction\n   * (see {@link bar#colorFunction})\n   * @param {number} [_=none]\n   * @returns {bar | number}\n   * @memberof bar\n   * @property\n   * by default colorFunction = colorFunction()\n   */\n  bar.colorFunction = function(_) { return arguments.length ? (colorFunction = _, bar) : colorFunction; };\n\n  /**\n   * Gets / sets the backgroundFill\n   * (see {@link bar#backgroundFill})\n   * @param {string} [_=none]\n   * @returns {bar | string}\n   * @memberof bar\n   * @property\n   * by default backgroundFill = 'transparent'\n   */\n  bar.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, bar) : backgroundFill; };\n  /**\n   * Gets / sets the namespace\n   * (see {@link bar#namespace})\n   * @param {string} [_=none]\n   * @returns {bar | string}\n   * @memberof bar\n   * @property\n   * by default namespace = 'd3sm-bar'\n   */\n  bar.namespace = function(_) { return arguments.length ? (namespace = _, bar) : namespace; };\n  /**\n   * Gets / sets the objectClass\n   * (see {@link bar#objectClass})\n   * @param {string} [_=none]\n   * @returns {bar | string}\n   * @memberof bar\n   * @property\n   * by default objectClass = 'tick-group'\n   */\n  bar.objectClass = function(_) { return arguments.length ? (objectClass = _, bar) : objectClass; };\n  /**\n   * Gets / sets the transitionDuration\n   * (see {@link bar#transitionDuration})\n   * @param {number} [_=none]\n   * @returns {bar | number}\n   * @memberof bar\n   * @property\n   * by default transitionDuration = 1000\n   */\n  bar.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, bar) : transitionDuration; };\n  /**\n   * Gets / sets the easeFunc\n   * (see {@link bar#easeFunc})\n   * @param {d3.ease} [_=none]\n   * @returns {bar | d3.ease}\n   * @memberof bar\n   * @property\n   * by default easeFunc = d3.easeExp\n   */\n  bar.easeFunc = function(_) { return arguments.length ? (easeFunc = _, bar) : easeFunc; };\n\n\n  /**\n   * Gets / sets the barKeys\n   * (see {@link bar#barKeys})\n   * @param {string[]} [_=none]\n   * @returns {bar | string[]}\n   * @memberof bar\n   * @property\n   * by default barKeys = undefined\n   */\n  bar.barKeys = function(_) { return arguments.length ? (barKeys = _, bar) : barKeys; };\n  /**\n   * Gets / sets the barValues\n   * (see {@link bar#barValues})\n   * @param {number[]} [_=none]\n   * @returns {bar | number[]}\n   * @memberof bar\n   * @property\n   * by default barValues = undefined\n   */\n  bar.barValues = function(_) { return arguments.length ? (barValues = _, bar) : barValues; };\n  /**\n   * Gets / sets the objectSize\n   * (see {@link bar#objectSize})\n   * @param {number} [_=none]\n   * @returns {bar | number}\n   * @memberof bar\n   * @property\n   * by default objectSize = undefined\n   */\n  bar.objectSize = function(_) { return arguments.length ? (objectSize = _, bar) : objectSize; };\n  /**\n   * Gets / sets the spacerSize\n   * (see {@link bar#spacerSize})\n   * @param {number} [_=none]\n   * @returns {bar | number}\n   * @memberof bar\n   * @property\n   * by default spacerSize = undefined\n   */\n  bar.spacerSize = function(_) { return arguments.length ? (spacerSize = _, bar) : spacerSize; };\n\n  /**\n   * Gets / sets the tooltip\n   * (see {@link bar#tooltip})\n   * @param {tooltip} [_=none]\n   * @returns {bar | tooltip}\n   * @memberof bar\n   * @property\n   * by default tooltip = tooltip()\n   */\n  bar.tooltip = function(_) { return arguments.length ? (tooltip = _, bar) : tooltip; };\n\n  bar.barPercent = function(_) { return arguments.length ? (barPercent = _, bar) : barPercent; };\n\n  function bar() {\n    // for convenience in handling orientation specific values\n    var horizontalQ = (orient == 'horizontal' || orient == 'bottom' || orient == 'top') ? true : false\n    var verticalQ = !horizontalQ\n\n    // background cliping rectangle\n    var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}\n    var container = setupContainer( selection, namespace, bgcpRect, backgroundFill );\n\n    // to prevent re-calculation and getters to be passed to axes\n    barKeys = d3.keys(data)\n    barValues = barKeys.map(valueExtractor)\n\n    // if grouping is undefined sort barKeys by sortingFunction\n    var ordered = (grouping == undefined) ? barKeys.sort(sortingFunction) : grouping\n    // ordered might be nested depending on grouping\n    barKeys = flatten(ordered)\n\n    var numberOfObjects = barKeys.length\n    var extent = [Math.min(...barValues) - domainPadding,Math.max(...barValues) + domainPadding];\n\n\n\n    // set the scale\n\n    scale.domain(extent).range(horizontalQ\n      ? [0,spaceY]\n      : orient == 'right'\n        ? [0, spaceX]\n        : [spaceX, 0]\n    )\n    var space = horizontalQ ? spaceX : spaceY\n    // calculate object size\n    objectSize =  (objectSize == undefined)\n    ? calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ)\n    : objectSize\n\n    // calculate spacer size if needed\n    spacerSize = (spacerSize == undefined)\n    ? calculateWidthOfSpacer(barKeys, space, objectSize, numberOfObjects, objectSpacer, overflowQ)\n    : spacerSize\n    // make the nested groups\n    var spacerFunction = groupingSpacer()\n    .horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects)\n    .objectClass(objectClass).objectSize(objectSize).spacerSize(spacerSize)\n    .transitionDuration(transitionDuration).easeFunc(easeFunc)\n    .namespace(namespace)\n    // safe default function\n    var defaultExit = spacerFunction.exitFunction()\n\n    spacerFunction.exitFunction(function(sel){\n      // use default to move objects off screen\n      // console.log(\"EXIT\", sel.nodes(), objectSize, scale(extent[1]))\n      if (objectSize == undefined) {console.log(sel.nodes(), objectSize)}\n      defaultExit(sel)\n      sel.selectAll('g').classed(\"to-remove\", true)\n      // shrink rectangles in addition\n      sel.selectAll('* > rect')\n      .transition().duration(transitionDuration)\n      .attr('transform', function(d, i) {\n        var\n        x = horizontalQ\n          ? 0\n          : 0\n        ,\n        y = verticalQ\n          ? 0\n          : scale(extent[1])\n        ,\n        t = 'translate('+x+','+y+')'\n        return t\n      })\n      .attr('width', horizontalQ ? objectSize : 0)\n      .attr('height', verticalQ ? objectSize : 0)\n      .remove()\n    })\n\n\n    // move stuff\n    spacerFunction(container, ordered, 0)\n\n\n\n\n\n    var parentIndexArray = []\n    container.selectAll('g:not(.to-remove).'+objectClass)\n    .each(function(d, i){parentIndexArray.push(Number(d3.select(this).attr('parent-index')))})\n\n\n    colorFunction = colorFunction.colorBy() == 'index'\n    ? colorFunction.dataExtent([0, Math.max(...parentIndexArray)])\n    : colorFunction.dataExtent(extent)\n\n\n\n    container.selectAll('g.'+objectClass+':not(.to-remove)').each(function(key, i) {\n      // console.log(key, scale(extent[1]) - scale(valueExtractor(key, i)))\n      var t = d3.select(this),\n      currentData = data[key],\n      value = valueExtractor(key, i),\n      i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'),\n      fillColor = colorFunction(key, value, i, 'fill'), // prevent duplicate computation\n      strokeColor = colorFunction(key, value, i,  'stroke')\n\n\n      var bar = safeSelect(t, 'rect', 'bar-rect')\n\n      if (bar.attr('transform') == undefined) {\n        bar.attr('transform', function(d, i) {\n          var\n          x = horizontalQ\n            ? 0\n            : 0\n          ,\n          y = verticalQ\n            ? 0\n            : scale(extent[1])\n          ,\n          t = 'translate('+x+','+y+')'\n          return t\n        })\n        .attr('width', horizontalQ ? objectSize : 0)\n        .attr('height', verticalQ ? objectSize : 0)\n\n      }\n\n\n      bar.transition().duration(transitionDuration).ease(easeFunc)\n      .attr('transform', function(d, i) {\n        var\n        x = horizontalQ\n          ? objectSize - objectSize * barPercent\n          : orient == 'right'\n            ? scale(extent[1]) - scale(value)\n            : objectSize - objectSize * barPercent\n          ,\n        y = verticalQ\n          ? objectSize - objectSize * barPercent\n          : scale(extent[1]) - scale(value)\n        ,\n        t = 'translate('+x+','+y+')'\n        return t\n      })\n      .attr('width', horizontalQ ? objectSize * barPercent : scale(value))\n      .attr('height', verticalQ ? objectSize * barPercent: scale(value))\n      .attr('fill', fillColor)\n      .attr('stroke', strokeColor)\n      .attr('stroke-width', barStrokeWidth)\n\n\n\n      t.on('mouseover', function(d, i){\n        container.selectAll('g.'+objectClass).style('opacity', 0.2)\n        t.style('opacity', 1)\n        bar.attr('stroke-width',barStrokeWidth*2)\n\n      })\n      t.on('mouseout', function(){\n        container.selectAll('g.'+objectClass).style('opacity', 1)\n        bar.attr('stroke-width', barStrokeWidth)\n      })\n    })\n\n    tooltip.selection(container.selectAll('.bar-rect'))\n    .data(data)\n\n    tooltip()\n\n  }\n  return bar\n}\n","import {hypenate, safeSelect} from './helpers';\nimport {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer, log} from './utils';\nimport {unique} from './array-functions';\nimport {groupingSpacer} from './grouping-spacer';\nimport {colorFunction as CF} from './color-function';\nimport {tooltip as TTip} from './tooltip';\n\n\n/**\n * Creates a bubbleHeatmap\n *\n * {@link https://sumneuron.gitlab.io/d3sm/demos/bubble-heatmap/index.html Demo}\n * @constructor bubbleHeatmap\n * @param {d3.selection} selection\n * @namespace bubbleHeatmap\n * @returns {function} bubbleHeatmap\n */\nfunction bubbleHeatmap( selection ) {\n  var\n  /**\n  * Data to plot. Assumed to be a object, where each key corresponds to a cell\n  * (see {@link bubbleHeatmap#data})\n  * @param {Object} [data=undefined]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  data,\n\n  /**\n  * Amount of horizontal space (in pixels) avaible to render the bubbleHeatmap in\n  * (see {@link bubbleHeatmap#spaceX})\n  * @param {number} [spaceX=undefined]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  spaceX,\n  /**\n  * Amount of vertical space (in pixels) avaible to render the bubbleHeatmap in\n  * (see {@link bubbleHeatmap.spaceY})\n  * @param {number} [spaceY=undefined]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  spaceY,\n\n  /**\n  * The internal key of the cell specifiying to which x axis key it belongs\n  * (see {@link bubbleHeatmap.xKey})\n  * @param {string} [xKey='x']\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  xKey = 'x',\n  /**\n  * The internal key of the cell specifiying to which y axis key it belongs\n  * (see {@link bubbleHeatmap.yKey})\n  * @param {string} [yKey='y']\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  yKey = 'y',\n  /**\n  * The internal key of the cell specifiying what value to use to determine the radius\n  * (see {@link bubbleHeatmap.rKey})\n  * @param {string} [rKey='r']\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  rKey = 'r',\n  /**\n  * The internal key of the cell specifiying what value to use to determine the color\n  * (see {@link bubbleHeatmap.vKey})\n  * @param {string} [vKey='v']\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  vKey = 'v',\n\n  /**\n  * Function for extracting the the value from xKey.\n  * (see {@link bubbleHeatmap.xExtractor})\n  * @param {function} [xExtractor=function(key, i) { return data[key][xKey] }]\n  * @returns {string}\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  xExtractor = function(key, i) {return data[key][xKey] },\n  /**\n  * Function for extracting the the value from yKey.\n  * (see {@link bubbleHeatmap.yExtractor})\n  * @param {function} [yExtractor=function(key, i) { return data[key][yKey] }]\n  * @returns {string}\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  yExtractor = function(key, i) { return data[key][yKey] },\n  /**\n  * Function for extracting the the value from rKey.\n  * (see {@link bubbleHeatmap.rExtractor})\n  * @param {function} [rExtractor=function(key, i) { return data[key][rKey] }]\n  * @returns {number}\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  rExtractor = function(key, i) { return data[key][rKey] },\n  /**\n  * Function for extracting the the value from vKey.\n  * (see {@link bubbleHeatmap.vExtractor})\n  * @param {function} [vExtractor=function(key, i) { return data[key][vKey] }]\n  * @returns {number}\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  vExtractor = function(key, i) { return data[key][vKey] },\n\n\n  /**\n  * Whether or not to allow bubbleHeatmap to render elements pass the main spatial dimension\n  * given the orientation (see {@link bubbleHeatmap#orient}), where {@link bubbleHeatmap#orient}=\"bottom\" or {@link bubbleHeatmap#orient}=\"top\"\n  * the main dimension is {@link bubbleHeatmap#spaceX} and where {@link bubbleHeatmap#orient}=\"left\" or {@link bubbleHeatmap#orient}=\"right\"\n  * the main dimension is {@link bubbleHeatmap#spaceY}\n  * @param {boolean} [overflowQ=false]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  overflowQ = false,\n\n  /**\n  * The scale for which the radius values should be transformed by\n  * @param {d3.scale} [scale=d3.scaleLinear]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  scale = d3.scaleLinear(),\n  /**\n  * The padding for the domain of the scale (see {@link bubbleHeatmap#scale})\n  * @param {number} [domainPadding=0.5]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  domainPadding = 0.5,\n\n  /**\n  * Default space for the spacer (percentage) of main dimension given the orientation\n  * (see {@link bubbleHeatmap#orient}), where {@link bubbleHeatmap#orient}=\"horizontal\"\n  * the main dimension is {@link bubbleHeatmap#spaceX} and where {@link bubbleHeatmap#orient}=\"vertical\"\n  * the main dimension is {@link bubbleHeatmap#spaceY} between bubbles\n  * @param {number} [objectSpacer=0.0]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  objectSpacer = 0.0,\n  /**\n  * The minimum size that an object can be\n  * @param {number} [minObjectSize=50]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  minObjectSize = 50,\n  /**\n  * The maximum size that an object can be\n  * @param {number} [maxObjectSize=100]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  maxObjectSize = 100,\n\n\n  /**\n  * The stroke width of the bubbles\n  * @param {number} [bubbleStrokeWidth=2]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  bubbleStrokeWidth = 2,\n  // colorFunc = colorFunction(),\n\n  /**\n  * Color of the background\n  * @param {string} [backgroundFill=\"transparent\"]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  backgroundFill = 'transparent',\n  /**\n  * Namespace for all items made by this instance of bubbleHeatmap\n  * @param {string} [namespace=\"d3sm-bubble\"]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  namespace = 'd3sm-bubble',\n  /**\n  * Class name for bubble container (<g> element)\n  * @param {string} [objectClass=\"bubble\"]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  objectClass = 'bubble',\n  /**\n  * Duration of all transitions of this element\n  * @param {number} [transitionDuration=1000]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  transitionDuration = 1000,\n  /**\n  * Easing function for transitions\n  * @param {d3.ease} [easeFunc=d3.easeExp]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  easeFunc = d3.easeExp,\n\n  /**\n  * Stores the keys of all the cells\n  * Calculated after bubbleHeatmap called.\n  * @param {string[]} [cellKeys=undefined]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  cellKeys,\n  /**\n  * Stores the list of unique xValues\n  * Calculated after bubbleHeatmap called.\n  * @param {string[]} [xValues=undefined]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  xValues,\n  /**\n  * Stores the list of unique yValues\n  * Calculated after bubbleHeatmap called.\n  * @param {string[]} [yValues=undefined]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  yValues,\n  /**\n  * Stores the list of unique rValues\n  * Calculated after bubbleHeatmap called.\n  * @param {string[]} [rValues=undefined]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  rValues,\n  /**\n  * Stores the list of unique vValues\n  * Calculated after bubbleHeatmap called.\n  * @param {string[]} [vValues=undefined]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  vValues,\n\n  xKeySortingFunction = function(a, b) { return xExtractor(a) - xExtractor(b) },\n  yKeySortingFunction = function(a, b) { return yExtractor(a) - yExtractor(b) },\n  rKeySortingFunction = function(a, b) { return rExtractor(a) - rExtractor(b) },\n  vKeySortingFunction = function(a, b) { return vExtractor(a) - vExtractor(b) },\n\n  /**\n  * Instance of ColorFunction with .colorBy set to 'category'\n  * @function colorFunction\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  colorFunction = CF().colorBy('value'),\n  /**\n  * Instance of Tooltip\n  * @function tooltip\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  tooltip = TTip(),\n\n  /**\n  * store the size the bubble could be in the x dimension\n  * the actuall size of the bubble will be the min of xSize and {@link bubbleHeatmap#ySize}\n  * Calculated after bubbleHeatmap called.\n  * @param {string[]} [xSize=undefined]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  xSize,\n  /**\n  * store the size of the spacer in the x dimension\n  * Calculated after bubbleHeatmap called.\n  * @param {string[]} [xSpacerSize=undefined]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  xSpacerSize,\n\n  /**\n  * store the size the bubble could be in the y dimension\n  * the actuall size of the bubble will be the min of xSize and {@link bubbleHeatmap#xSize}\n  * Calculated after bubbleHeatmap called.\n  * @param {string[]} [ySize=undefined]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  ySize,\n  /**\n  * store the size of the spacer in the y dimension.\n  * Calculated after bubbleHeatmap called.\n  * @param {string[]} [xSpacerSize=undefined]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  ySpacerSize\n\n  /**\n   * Gets or sets the selection in which items are manipulated\n   * @param {d3.selection} [_=none]\n   * @returns {bubbleHeatmap | d3.selection}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default selection = selection\n   */\n  bhm.selection = function(_) { return arguments.length ? (selection = _, bhm) : selection; }\n  /**\n   * Gets or sets the data\n   * (see {@link bubbleHeatmap#data})\n   * @param {number} [_=none]\n   * @returns {bubbleHeatmap | object}\n   * @memberof bubbleHeatmap\n   * @property\n   */\n  bhm.data = function(_) { return arguments.length ? (data = _, bhm) : data; }\n  // bhm.orient = function(_) { return arguments.length ? (orient = _, bhm) : orient; }\n  /**\n   * Gets or sets the amount of horizontal space in which items are manipulated\n   * (see {@link bubbleHeatmap#spaceX})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {bubbleHeatmap | number}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default spaceX = undefined\n   */\n  bhm.spaceX = function(_) { return arguments.length ? (spaceX = _, bhm) : spaceX; }\n  /**\n   * Gets or sets the amount of vertical space in which items are manipulated\n   * (see {@link bubbleHeatmap#spaceY})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {bubbleHeatmap | number}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default spaceY = undefined\n   */\n  bhm.spaceY = function(_) { return arguments.length ? (spaceY = _, bhm) : spaceY; }\n\n  /**\n   * Gets or sets the xKey\n   * (see {@link bubbleHeatmap#xKey})\n   * @param {string} [_=none]\n   * @returns {bubbleHeatmap | string}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default xKey = 'x'\n   */\n  bhm.xKey = function(_) { return arguments.length ? (xKey = _, bhm) : xKey; }\n  /**\n   * Gets or sets the yKey\n   * (see {@link bubbleHeatmap#yKey})\n   * @param {string} [_=none]\n   * @returns {bubbleHeatmap | string}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default yKey = 'y'\n   */\n  bhm.yKey = function(_) { return arguments.length ? (yKey = _, bhm) : yKey; }\n  /**\n   * Gets or sets the rKey\n   * (see {@link bubbleHeatmap#rKey})\n   * @param {string} [_=none]\n   * @returns {bubbleHeatmap | string}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default rKey = 'r'\n   */\n  bhm.rKey = function(_) { return arguments.length ? (rKey = _, bhm) : rKey; }\n  /**\n   * Gets or sets the vKey\n   * (see {@link bubbleHeatmap#vKey})\n   * @param {string} [_=none]\n   * @returns {bubbleHeatmap | string}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default vKey = 'y'\n   */\n  bhm.vKey = function(_) { return arguments.length ? (vKey = _, bhm) : vKey; }\n\n  /**\n   * Gets or sets the cellKeys\n   * (see {@link bubbleHeatmap#cellKeys})\n   * @param {string[]} [_=none]\n   * @returns {bubbleHeatmap | string[]}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default cellKeys = undefined\n   */\n  bhm.cellKeys = function(_) { return arguments.length ? (cellKeys = _, bhm) : cellKeys; }\n  /**\n   * Gets or sets the xValues\n   * (see {@link bubbleHeatmap#xValues})\n   * @param {string[]} [_=none]\n   * @returns {bubbleHeatmap | string[]}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default xValues = undefined\n   */\n  bhm.xValues = function(_) { return arguments.length ? (xValues = _, bhm) : xValues; }\n  /**\n   * Gets or sets the yValues\n   * (see {@link bubbleHeatmap#yValues})\n   * @param {string[]} [_=none]\n   * @returns {bubbleHeatmap | string[]}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default yValues = undefined\n   */\n  bhm.yValues = function(_) { return arguments.length ? (yValues = _, bhm) : yValues; }\n  /**\n   * Gets or sets the rValues\n   * (see {@link bubbleHeatmap#rValues})\n   * @param {number[]} [_=none]\n   * @returns {bubbleHeatmap | number[]}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default rValues = undefined\n   */\n  bhm.rValues = function(_) { return arguments.length ? (rValues = _, bhm) : rValues; }\n  /**\n   * Gets or sets the vValues\n   * (see {@link bubbleHeatmap#vValues})\n   * @param {number[]} [_=none]\n   * @returns {bubbleHeatmap | number[]}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default vValues = undefined\n   */\n  bhm.vValues = function(_) { return arguments.length ? (vValues = _, bhm) : vValues; }\n\n\n  /**\n   * Gets or sets the xExtractor\n   * (see {@link bubbleHeatmap#xExtractor})\n   * @param {function} [_=none]\n   * @returns {bubbleHeatmap | function}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default xExtractor = undefined\n   */\n  bhm.xExtractor = function(_) { return arguments.length ? (xExtractor = _, bhm) : xExtractor; }\n  /**\n   * Gets or sets the yExtractor\n   * (see {@link bubbleHeatmap#yExtractor})\n   * @param {function} [_=none]\n   * @returns {bubbleHeatmap | function}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default yExtractor = undefined\n   */\n  bhm.yExtractor = function(_) { return arguments.length ? (yExtractor = _, bhm) : yExtractor; }\n  /**\n   * Gets or sets the rExtractor\n   * (see {@link bubbleHeatmap#rExtractor})\n   * @param {function} [_=none]\n   * @returns {bubbleHeatmap | function}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default rExtractor = undefined\n   */\n  bhm.rExtractor = function(_) { return arguments.length ? (rExtractor = _, bhm) : rExtractor; }\n  /**\n   * Gets or sets the vExtractor\n   * (see {@link bubbleHeatmap#vExtractor})\n   * @param {function} [_=none]\n   * @returns {bubbleHeatmap | function}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default vExtractor = undefined\n   */\n  bhm.vExtractor = function(_) { return arguments.length ? (vExtractor = _, bhm) : vExtractor; }\n\n  /**\n   * Gets / sets whether or not bubbleHeatmap is allowed to go beyond specified dimensions\n   * (see {@link bubbleHeatmap#spaceX})\n   * @param {boolean} [_=none]\n   * @returns {bubbleHeatmap | boolean}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default overflowQ = false\n   */\n  bhm.overflowQ = function(_) { return arguments.length ? (overflowQ = _, bhm) : overflowQ; }\n  /**\n   * Gets / sets the scale for which the radius of bubbles should be transformed by\n   * (see {@link bubbleHeatmap#scale})\n   * @param {d3.scale} [_=none]\n   * @returns {bubbleHeatmap | d3.scale}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default scale = d3.scaleLinear()\n   */\n  bhm.scale = function(_) { return arguments.length ? (scale = _, bhm) : scale; };\n  /**\n   * Gets / sets the padding for the domain of the scale\n   * (see {@link bubbleHeatmap#domainPadding})\n   * @param {number} [_=none]\n   * @returns {bubbleHeatmap | number}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default domainPadding = 0.5\n   */\n  bhm.domainPadding = function(_) { return arguments.length ? (domainPadding = _, bhm) : domainPadding; };\n  /**\n   * Gets / sets objectSpacer\n   * (see {@link bubbleHeatmap#objectSpacer})\n   * @param {number} [_=none]\n   * @returns {bubbleHeatmap | number}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default objectSpacer = 0.0\n   */\n  bhm.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, objectSpacer) : data; }\n  /**\n   * Gets / sets the minObjectSize\n   * (see {@link bubbleHeatmap#minObjectSize})\n   * @param {number} [_=none]\n   * @returns {bubbleHeatmap | number}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default minObjectSize = 50\n   */\n  bhm.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, bhm) : minObjectSize; }\n  /**\n   * Gets / sets the maxObjectSize\n   * (see {@link bubbleHeatmap#maxObjectSize})\n   * @param {number} [_=none]\n   * @returns {bubbleHeatmap | number}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default maxObjectSize = 100\n   */\n  bhm.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, bhm) : maxObjectSize; }\n  /**\n   * Gets / sets the bubbleStrokeWidth\n   * (see {@link bubbleHeatmap#bubbleStrokeWidth})\n   * @param {number} [_=none]\n   * @returns {bubbleHeatmap | number}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default bubbleStrokeWidth = 2\n   */\n  bhm.bubbleStrokeWidth = function(_) { return arguments.length ? (bubbleStrokeWidth = _, bhm) : bubbleStrokeWidth; }\n  /**\n   * Gets / sets the backgroundFill\n   * (see {@link bubbleHeatmap#backgroundFill})\n   * @param {string} [_=none]\n   * @returns {bubbleHeatmap | string}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default backgroundFill = 'transparent'\n   */\n  bhm.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, bhm) : backgroundFill; }\n  /**\n   * Gets / sets the namespace\n   * (see {@link bubbleHeatmap#namespace})\n   * @param {string} [_=none]\n   * @returns {bubbleHeatmap | string}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default namespace = 'd3sm-bubbleHeatmap'\n   */\n  bhm.namespace = function(_) { return arguments.length ? (namespace = _, bhm) : namespace; }\n  /**\n   * Gets / sets the objectClass\n   * (see {@link bubbleHeatmap#objectClass})\n   * @param {string} [_=none]\n   * @returns {bubbleHeatmap | string}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default objectClass = 'tick-group'\n   */\n  bhm.objectClass = function(_) { return arguments.length ? (objectClass = _, bhm) : objectClass; }\n  /**\n   * Gets / sets the transitionDuration\n   * (see {@link bubbleHeatmap#transitionDuration})\n   * @param {number} [_=none]\n   * @returns {bubbleHeatmap | number}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default transitionDuration = 1000\n   */\n  bhm.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, bhm) : transitionDuration; }\n  /**\n   * Gets / sets the easeFunc\n   * (see {@link bubbleHeatmap#easeFunc})\n   * @param {d3.ease} [_=none]\n   * @returns {bubbleHeatmap | d3.ease}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default easeFunc = d3.easeExp\n   */\n  bhm.easeFunc = function(_) { return arguments.length ? (easeFunc = _, bhm) : easeFunc; }\n\n  /**\n   * Gets / sets the tooltip\n   * (see {@link bubbleHeatmap#tooltip})\n   * @param {tooltip} [_=none]\n   * @returns {bubbleHeatmap | tooltip}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default tooltip = tooltip()\n   */\n  bhm.tooltip = function(_) { return arguments.length ? (tooltip = _, bhm) : tooltip; }\n\n  /**\n   * Gets / sets the colorFunction\n   * (see {@link bubbleHeatmap#colorFunction})\n   * @param {colorFunction} [_=none]\n   * @returns {bubbleHeatmap | colorFunction}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default colorFunction = colorFunction()\n   */\n  bhm.colorFunction = function(_) { return arguments.length ? (colorFunction = _, bhm) : colorFunction; }\n\n  /**\n   * Gets / sets the xSize\n   * (see {@link bubbleHeatmap#xSize})\n   * @param {number} [_=none]\n   * @returns {bubbleHeatmap | number}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default xSize = undefined\n   */\n  bhm.xSize = function(_) { return arguments.length ? (xSize = _, bhm) : xSize; }\n  /**\n   * Gets / sets the xSpacerSize\n   * (see {@link bubbleHeatmap#xSpacerSize})\n   * @param {number} [_=none]\n   * @returns {bubbleHeatmap | number}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default xSpacerSize = undefined\n   */\n  bhm.xSpacerSize = function(_) { return arguments.length ? (xSpacerSize = _, bhm) : xSpacerSize; }\n  /**\n   * Gets / sets the ySize\n   * (see {@link bubbleHeatmap#ySize})\n   * @param {number} [_=none]\n   * @returns {bubbleHeatmap | number}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default ySize = undefined\n   */\n  bhm.ySize = function(_) { return arguments.length ? (ySize = _, bhm) : ySize; }\n  /**\n   * Gets / sets the ySpacerSize\n   * (see {@link bubbleHeatmap#ySpacerSize})\n   * @param {number} [_=none]\n   * @returns {bubbleHeatmap | number}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default ySpacerSize = undefined\n   */\n  bhm.ySpacerSize = function(_) { return arguments.length ? (ySpacerSize = _, bhm) : ySpacerSize; }\n  // bhm.yKeySortingFunction = function(_) { return arguments.length ? (yKeySortingFunction = _, bhm) : yKeySortingFunction; }\n  // bhm.xKeySortingFunction = function(_) { return arguments.length ? (xKeySortingFunction = _, bhm) : xKeySortingFunction; }\n\n\n\n  function bhm() {\n    var horizontalQ = true; // no orientation in heatmaps. Aids as placeholder in functions that take orient as a parameter\n    var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}\n    var container = setupContainer( selection, namespace, bgcpRect, backgroundFill );\n\n    cellKeys = d3.keys(data);\n    cellKeys.sort(function(a, b){ return xKeySortingFunction(a, b) || yKeySortingFunction(a, b) })\n    log('bubbleHeatmap', 'cells are sorted by', cellKeys)\n\n\n\n    xValues = unique(cellKeys.map(xExtractor));\n    yValues = unique(cellKeys.map(yExtractor));\n    rValues = unique(cellKeys.map(rExtractor));\n    vValues = unique(cellKeys.map(vExtractor));\n    log('bubbleHeatmap', 'x and y keys are', {x: xValues, y:yValues})\n\n\n    var xDim = xValues.length, yDim = yValues.length;\n\n\n    var extent = [Math.min(...rValues) - domainPadding,Math.max(...rValues) + domainPadding];\n\n\n    ySize = calculateWidthOfObject(spaceY, yDim, minObjectSize, maxObjectSize, objectSpacer, overflowQ)\n    xSize = calculateWidthOfObject(spaceX, xDim, minObjectSize, maxObjectSize, objectSpacer, overflowQ)\n    ySpacerSize = calculateWidthOfSpacer(yValues, spaceY, ySize, yDim, objectSpacer, overflowQ)\n    xSpacerSize = calculateWidthOfSpacer(xValues, spaceX, xSize, xDim, objectSpacer, overflowQ)\n    log('bubbleHeatmap', 'size of', {x: xSize, y: ySize})\n\n\n    scale.domain(extent).range([Math.min(minObjectSize/2,   Math.min(ySize, xSize)/2), Math.min(ySize, xSize)/2])\n\n    var ySpacer = groupingSpacer()\n    .horizontalQ(false)\n    .moveby('category').numberOfObjects(yDim)\n    .objectClass(hypenate(objectClass, 'row'))\n    .objectSize(ySize).spacerSize(ySpacerSize)\n    .transitionDuration(transitionDuration).easeFunc(easeFunc)\n    .namespace('row')\n\n    var xSpacer = groupingSpacer()\n    .horizontalQ(true)\n    .moveby('category').numberOfObjects(xDim)\n    .objectClass(objectClass)\n    .objectSize(xSize).spacerSize(xSpacerSize)\n    .transitionDuration(transitionDuration).easeFunc(easeFunc)\n\n\n    ySpacer(container, yValues, 0)\n    container.selectAll('g.'+hypenate(objectClass, 'row'))\n    .each(function(d, i){\n      xSpacer(d3.select(this), xValues, 0)\n    })\n    var cells = container.selectAll('g:not(.to-remove).'+objectClass).data(cellKeys);\n\n    var parentIndexArray = []\n    cells.each(function(d, i){ parentIndexArray.push(Number(d3.select(this).attr('parent-index'))) })\n\n\n    colorFunction = colorFunction.colorBy() == 'index'\n    ? colorFunction.dataExtent([0, Math.max(...parentIndexArray)])\n    : colorFunction.dataExtent([0, Math.max(...vValues)])\n\n    cells.each(function(key, i) {\n      log('bubbleHeatmap', 'each cell', {key: key, index: i, node: d3.select(this).node()})\n\n      var t = d3.select(this),\n      currentData = data[key],\n      value = vExtractor(key, i),\n      radius= rExtractor(key, i),\n      i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'),\n      fillColor = colorFunction(key, value, i, 'fill'), // prevent duplicate computation\n      strokeColor = colorFunction(key, value, i,  'stroke')\n\n      log('bubbleHeatmap', 'radius',{radius: radius, scaled: scale(radius), extent: extent, range:scale.range()})\n\n      var c = safeSelect(t, 'circle', hypenate(objectClass,'circle'))\n      c.attr('cx', xSize / 2)\n      .attr('cy', ySize / 2 )\n      .attr('r', scale(radius))\n      .attr('fill', fillColor)\n      .attr('stroke', strokeColor)\n      .attr('stroke-width', bubbleStrokeWidth)\n\n    })\n\n    tooltip.selection(cells.selectAll('circle.'+hypenate(objectClass, 'circle')))\n    .data(data)\n    // .keys(['r', 'v'])\n    // .header(function(d, i){return hypenate(data[d][xKey], data[d][yKey]) })\n\n    tooltip()\n\n\n  }\n\n  return bhm;\n}\n\nexport {bubbleHeatmap}\n","import {hypenate, safeSelect} from './helpers';\nimport {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer, log} from './utils';\nimport {unique} from './array-functions';\nimport {groupingSpacer} from './grouping-spacer';\nimport {colorFunction as CF} from './color-function';\nimport {tooltip as TTip} from './tooltip';\n\n\n/**\n * Creates a heatmap\n *\n * {@link https://sumneuron.gitlab.io/d3sm/demos/heatmap-heatmap/index.html Demo}\n * @constructor heatmap\n * @param {d3.selection} selection\n * @namespace heatmap\n * @returns {function} heatmap\n */\nfunction heatmap( selection ) {\n  var\n  /**\n  * Data to plot. Assumed to be a object, where each key corresponds to a cell\n  * (see {@link heatmap#data})\n  * @param {Object} [data=undefined]\n  * @memberof heatmap#\n  * @property\n  */\n  data,\n\n  /**\n  * Amount of horizontal space (in pixels) avaible to render the heatmap in\n  * (see {@link heatmap#spaceX})\n  * @param {number} [spaceX=undefined]\n  * @memberof heatmap#\n  * @property\n  */\n  spaceX,\n  /**\n  * Amount of vertical space (in pixels) avaible to render the heatmap in\n  * (see {@link heatmap.spaceY})\n  * @param {number} [spaceY=undefined]\n  * @memberof heatmap#\n  * @property\n  */\n  spaceY,\n\n  /**\n  * The internal key of the cell specifiying to which x axis key it belongs\n  * (see {@link heatmap.xKey})\n  * @param {string} [xKey='x']\n  * @memberof heatmap#\n  * @property\n  */\n  xKey = 'x',\n  /**\n  * The internal key of the cell specifiying to which y axis key it belongs\n  * (see {@link heatmap.yKey})\n  * @param {string} [yKey='y']\n  * @memberof heatmap#\n  * @property\n  */\n  yKey = 'y',\n\n  /**\n  * The internal key of the cell specifiying what value to use to determine the color\n  * (see {@link heatmap.vKey})\n  * @param {string} [vKey='v']\n  * @memberof heatmap#\n  * @property\n  */\n  vKey = 'v',\n\n  /**\n  * Function for extracting the the value from xKey.\n  * (see {@link heatmap.xExtractor})\n  * @param {function} [xExtractor=function(key, i) { return data[key][xKey] }]\n  * @returns {string}\n  * @memberof heatmap#\n  * @property\n  */\n  xExtractor = function(key, i) {return data[key][xKey] },\n  /**\n  * Function for extracting the the value from yKey.\n  * (see {@link heatmap.yExtractor})\n  * @param {function} [yExtractor=function(key, i) { return data[key][yKey] }]\n  * @returns {string}\n  * @memberof heatmap#\n  * @property\n  */\n  yExtractor = function(key, i) { return data[key][yKey] },\n\n  /**\n  * Function for extracting the the value from vKey.\n  * (see {@link heatmap.vExtractor})\n  * @param {function} [vExtractor=function(key, i) { return data[key][vKey] }]\n  * @returns {number}\n  * @memberof heatmap#\n  * @property\n  */\n  vExtractor = function(key, i) { return data[key][vKey] },\n\n\n  /**\n  * Whether or not to allow heatmap to render elements pass the main spatial dimension\n  * given the orientation (see {@link heatmap#orient}), where {@link heatmap#orient}=\"bottom\" or {@link heatmap#orient}=\"top\"\n  * the main dimension is {@link heatmap#spaceX} and where {@link heatmap#orient}=\"left\" or {@link heatmap#orient}=\"right\"\n  * the main dimension is {@link heatmap#spaceY}\n  * @param {boolean} [overflowQ=false]\n  * @memberof heatmap#\n  * @property\n  */\n  overflowQ = false,\n\n  /**\n  * Default space for the spacer (percentage) of main dimension given the orientation\n  * (see {@link heatmap#orient}), where {@link heatmap#orient}=\"horizontal\"\n  * the main dimension is {@link heatmap#spaceX} and where {@link heatmap#orient}=\"vertical\"\n  * the main dimension is {@link heatmap#spaceY} between bubbles\n  * @param {number} [objectSpacer=0.0]\n  * @memberof heatmap#\n  * @property\n  */\n  objectSpacer = 0.0,\n  /**\n  * The minimum size that an object can be in the y dimension\n  * @param {number} [minObjectSize=50]\n  * @memberof heatmap#\n  * @property\n  */\n  yMinObjectSize = 50,\n  /**\n  * The minimum size that an object can be in the x dimension\n  * @param {number} [minObjectSize=50]\n  * @memberof heatmap#\n  * @property\n  */\n  xMinObjectSize = 50,\n  /**\n  * The maximum size that an object can be in the x dimension\n  * @param {number} [maxObjectSize=100]\n  * @memberof heatmap#\n  * @property\n  */\n  xMaxObjectSize = 100,\n  /**\n  * The maximum size that an object can be in the y dimension\n  * @param {number} [maxObjectSize=100]\n  * @memberof heatmap#\n  * @property\n  */\n  yMaxObjectSize = 100,\n\n\n  /**\n  * The stroke width of the bubbles\n  * @param {number} [objectStrokeWidth=2]\n  * @memberof heatmap#\n  * @property\n  */\n  objectStrokeWidth = 2,\n  // colorFunc = colorFunction(),\n\n  /**\n  * Color of the background\n  * @param {string} [backgroundFill=\"transparent\"]\n  * @memberof heatmap#\n  * @property\n  */\n  backgroundFill = 'transparent',\n  /**\n  * Namespace for all items made by this instance of heatmap\n  * @param {string} [namespace=\"d3sm-heatmap\"]\n  * @memberof heatmap#\n  * @property\n  */\n  namespace = 'd3sm-heatmap',\n  /**\n  * Class name for heatmap container (<g> element)\n  * @param {string} [objectClass=\"heatmap\"]\n  * @memberof heatmap#\n  * @property\n  */\n  objectClass = 'heatmap',\n  /**\n  * Duration of all transitions of this element\n  * @param {number} [transitionDuration=1000]\n  * @memberof heatmap#\n  * @property\n  */\n  transitionDuration = 1000,\n  /**\n  * Easing function for transitions\n  * @param {d3.ease} [easeFunc=d3.easeExp]\n  * @memberof heatmap#\n  * @property\n  */\n  easeFunc = d3.easeExp,\n\n  /**\n  * Stores the keys of all the cells\n  * Calculated after heatmap called.\n  * @param {string[]} [cellKeys=undefined]\n  * @memberof heatmap#\n  * @property\n  */\n  cellKeys,\n  /**\n  * Stores the list of unique xValues\n  * Calculated after heatmap called.\n  * @param {string[]} [xValues=undefined]\n  * @memberof heatmap#\n  * @property\n  */\n  xValues,\n  /**\n  * Stores the list of unique yValues\n  * Calculated after heatmap called.\n  * @param {string[]} [yValues=undefined]\n  * @memberof heatmap#\n  * @property\n  */\n  yValues,\n  /**\n  * Stores the list of unique vValues\n  * Calculated after heatmap called.\n  * @param {string[]} [vValues=undefined]\n  * @memberof heatmap#\n  * @property\n  */\n  vValues,\n\n  xKeySortingFunction = function(a, b) { return xValues.indexOf(xExtractor(a)) - xValues.indexOf(xExtractor(b)) },\n  yKeySortingFunction = function(a, b) { return yValues.indexOf(yExtractor(a)) - yValues.indexOf(yExtractor(b)) },\n  vKeySortingFunction = function(a, b) { return vValues.indexOf(vExtractor(a)) - yValues.indexOf(vExtractor(b)) },\n\n  /**\n  * Instance of ColorFunction with .colorBy set to 'category'\n  * @function colorFunction\n  * @memberof heatmap#\n  * @property\n  */\n  colorFunction = CF().colorBy('category'),\n  /**\n  * Instance of Tooltip\n  * @function tooltip\n  * @memberof heatmap#\n  * @property\n  */\n  tooltip = TTip(),\n\n  /**\n  * store the size the heatmap could be in the x dimension\n  * the actuall size of the heatmap will be the min of xSize and {@link heatmap#ySize}\n  * Calculated after heatmap called.\n  * @param {string[]} [xSize=undefined]\n  * @memberof heatmap#\n  * @property\n  */\n  xSize,\n  /**\n  * store the size of the spacer in the x dimension\n  * Calculated after heatmap called.\n  * @param {string[]} [xSpacerSize=undefined]\n  * @memberof heatmap#\n  * @property\n  */\n  xSpacerSize,\n\n  /**\n  * store the size the heatmap could be in the y dimension\n  * the actuall size of the heatmap will be the min of xSize and {@link heatmap#xSize}\n  * Calculated after heatmap called.\n  * @param {string[]} [ySize=undefined]\n  * @memberof heatmap#\n  * @property\n  */\n  ySize,\n  /**\n  * store the size of the spacer in the y dimension.\n  * Calculated after heatmap called.\n  * @param {string[]} [xSpacerSize=undefined]\n  * @memberof heatmap#\n  * @property\n  */\n  ySpacerSize\n\n  /**\n   * Gets or sets the selection in which items are manipulated\n   * @param {d3.selection} [_=none]\n   * @returns {heatmap | d3.selection}\n   * @memberof heatmap\n   * @property\n   * by default selection = selection\n   */\n  hm.selection = function(_) { return arguments.length ? (selection = _, hm) : selection; }\n  /**\n   * Gets or sets the data\n   * (see {@link heatmap#data})\n   * @param {number} [_=none]\n   * @returns {heatmap | object}\n   * @memberof heatmap\n   * @property\n   */\n  hm.data = function(_) { return arguments.length ? (data = _, hm) : data; }\n  // hm.orient = function(_) { return arguments.length ? (orient = _, hm) : orient; }\n  /**\n   * Gets or sets the amount of horizontal space in which items are manipulated\n   * (see {@link heatmap#spaceX})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {heatmap | number}\n   * @memberof heatmap\n   * @property\n   * by default spaceX = undefined\n   */\n  hm.spaceX = function(_) { return arguments.length ? (spaceX = _, hm) : spaceX; }\n  /**\n   * Gets or sets the amount of vertical space in which items are manipulated\n   * (see {@link heatmap#spaceY})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {heatmap | number}\n   * @memberof heatmap\n   * @property\n   * by default spaceY = undefined\n   */\n  hm.spaceY = function(_) { return arguments.length ? (spaceY = _, hm) : spaceY; }\n\n  /**\n   * Gets or sets the xKey\n   * (see {@link heatmap#xKey})\n   * @param {string} [_=none]\n   * @returns {heatmap | string}\n   * @memberof heatmap\n   * @property\n   * by default xKey = 'x'\n   */\n  hm.xKey = function(_) { return arguments.length ? (xKey = _, hm) : xKey; }\n  /**\n   * Gets or sets the yKey\n   * (see {@link heatmap#yKey})\n   * @param {string} [_=none]\n   * @returns {heatmap | string}\n   * @memberof heatmap\n   * @property\n   * by default yKey = 'y'\n   */\n  hm.yKey = function(_) { return arguments.length ? (yKey = _, hm) : yKey; }\n\n  /**\n   * Gets or sets the vKey\n   * (see {@link heatmap#vKey})\n   * @param {string} [_=none]\n   * @returns {heatmap | string}\n   * @memberof heatmap\n   * @property\n   * by default vKey = 'y'\n   */\n  hm.vKey = function(_) { return arguments.length ? (vKey = _, hm) : vKey; }\n\n  /**\n   * Gets or sets the cellKeys\n   * (see {@link heatmap#cellKeys})\n   * @param {string[]} [_=none]\n   * @returns {heatmap | string[]}\n   * @memberof heatmap\n   * @property\n   * by default cellKeys = undefined\n   */\n  hm.cellKeys = function(_) { return arguments.length ? (cellKeys = _, hm) : cellKeys; }\n  /**\n   * Gets or sets the xValues\n   * (see {@link heatmap#xValues})\n   * @param {string[]} [_=none]\n   * @returns {heatmap | string[]}\n   * @memberof heatmap\n   * @property\n   * by default xValues = undefined\n   */\n  hm.xValues = function(_) { return arguments.length ? (xValues = _, hm) : xValues; }\n  /**\n   * Gets or sets the yValues\n   * (see {@link heatmap#yValues})\n   * @param {string[]} [_=none]\n   * @returns {heatmap | string[]}\n   * @memberof heatmap\n   * @property\n   * by default yValues = undefined\n   */\n  hm.yValues = function(_) { return arguments.length ? (yValues = _, hm) : yValues; }\n  /**\n   * Gets or sets the vValues\n   * (see {@link heatmap#vValues})\n   * @param {number[]} [_=none]\n   * @returns {heatmap | number[]}\n   * @memberof heatmap\n   * @property\n   * by default vValues = undefined\n   */\n  hm.vValues = function(_) { return arguments.length ? (vValues = _, hm) : vValues; }\n\n\n  /**\n   * Gets or sets the xExtractor\n   * (see {@link heatmap#xExtractor})\n   * @param {function} [_=none]\n   * @returns {heatmap | function}\n   * @memberof heatmap\n   * @property\n   * by default xExtractor = undefined\n   */\n  hm.xExtractor = function(_) { return arguments.length ? (xExtractor = _, hm) : xExtractor; }\n  /**\n   * Gets or sets the yExtractor\n   * (see {@link heatmap#yExtractor})\n   * @param {function} [_=none]\n   * @returns {heatmap | function}\n   * @memberof heatmap\n   * @property\n   * by default yExtractor = undefined\n   */\n  hm.yExtractor = function(_) { return arguments.length ? (yExtractor = _, hm) : yExtractor; }\n  /**\n   * Gets or sets the vExtractor\n   * (see {@link heatmap#vExtractor})\n   * @param {function} [_=none]\n   * @returns {heatmap | function}\n   * @memberof heatmap\n   * @property\n   * by default vExtractor = undefined\n   */\n  hm.vExtractor = function(_) { return arguments.length ? (vExtractor = _, hm) : vExtractor; }\n\n  /**\n   * Gets / sets whether or not heatmap is allowed to go beyond specified dimensions\n   * (see {@link heatmap#spaceX})\n   * @param {boolean} [_=none]\n   * @returns {heatmap | boolean}\n   * @memberof heatmap\n   * @property\n   * by default overflowQ = false\n   */\n  hm.overflowQ = function(_) { return arguments.length ? (overflowQ = _, hm) : overflowQ; }\n  /**\n   * Gets / sets objectSpacer\n   * (see {@link heatmap#objectSpacer})\n   * @param {number} [_=none]\n   * @returns {heatmap | number}\n   * @memberof heatmap\n   * @property\n   * by default objectSpacer = 0.0\n   */\n  hm.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, objectSpacer) : data; }\n  /**\n   * Gets / sets the minObjectSize\n   * (see {@link heatmap#minObjectSize})\n   * @param {number} [_=none]\n   * @returns {heatmap | number}\n   * @memberof heatmap\n   * @property\n   * by default minObjectSize = 50\n   */\n  hm.yMinObjectSize = function(_) { return arguments.length ? (yMinObjectSize = _, hm) : yMinObjectSize; }\n  /**\n   * Gets / sets the maxObjectSize\n   * (see {@link heatmap#maxObjectSize})\n   * @param {number} [_=none]\n   * @returns {heatmap | number}\n   * @memberof heatmap\n   * @property\n   * by default maxObjectSize = 100\n   */\n  hm.yMaxObjectSize = function(_) { return arguments.length ? (yMaxObjectSize = _, hm) : yMaxObjectSize; }\n  /**\n   * Gets / sets the minObjectSize\n   * (see {@link heatmap#minObjectSize})\n   * @param {number} [_=none]\n   * @returns {heatmap | number}\n   * @memberof heatmap\n   * @property\n   * by default minObjectSize = 50\n   */\n  hm.xMinObjectSize = function(_) { return arguments.length ? (xMinObjectSize = _, hm) : xMinObjectSize; }\n  /**\n   * Gets / sets the maxObjectSize\n   * (see {@link heatmap#maxObjectSize})\n   * @param {number} [_=none]\n   * @returns {heatmap | number}\n   * @memberof heatmap\n   * @property\n   * by default maxObjectSize = 100\n   */\n  hm.xMaxObjectSize = function(_) { return arguments.length ? (xMaxObjectSize = _, hm) : xMaxObjectSize; }\n  /**\n   * Gets / sets the objectStrokeWidth\n   * (see {@link heatmap#objectStrokeWidth})\n   * @param {number} [_=none]\n   * @returns {heatmap | number}\n   * @memberof heatmap\n   * @property\n   * by default objectStrokeWidth = 2\n   */\n  hm.objectStrokeWidth = function(_) { return arguments.length ? (objectStrokeWidth = _, hm) : objectStrokeWidth; }\n  /**\n   * Gets / sets the backgroundFill\n   * (see {@link heatmap#backgroundFill})\n   * @param {string} [_=none]\n   * @returns {heatmap | string}\n   * @memberof heatmap\n   * @property\n   * by default backgroundFill = 'transparent'\n   */\n  hm.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, hm) : backgroundFill; }\n  /**\n   * Gets / sets the namespace\n   * (see {@link heatmap#namespace})\n   * @param {string} [_=none]\n   * @returns {heatmap | string}\n   * @memberof heatmap\n   * @property\n   * by default namespace = 'd3sm-heatmap'\n   */\n  hm.namespace = function(_) { return arguments.length ? (namespace = _, hm) : namespace; }\n  /**\n   * Gets / sets the objectClass\n   * (see {@link heatmap#objectClass})\n   * @param {string} [_=none]\n   * @returns {heatmap | string}\n   * @memberof heatmap\n   * @property\n   * by default objectClass = 'tick-group'\n   */\n  hm.objectClass = function(_) { return arguments.length ? (objectClass = _, hm) : objectClass; }\n  /**\n   * Gets / sets the transitionDuration\n   * (see {@link heatmap#transitionDuration})\n   * @param {number} [_=none]\n   * @returns {heatmap | number}\n   * @memberof heatmap\n   * @property\n   * by default transitionDuration = 1000\n   */\n  hm.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, hm) : transitionDuration; }\n  /**\n   * Gets / sets the easeFunc\n   * (see {@link heatmap#easeFunc})\n   * @param {d3.ease} [_=none]\n   * @returns {heatmap | d3.ease}\n   * @memberof heatmap\n   * @property\n   * by default easeFunc = d3.easeExp\n   */\n  hm.easeFunc = function(_) { return arguments.length ? (easeFunc = _, hm) : easeFunc; }\n\n  /**\n   * Gets / sets the tooltip\n   * (see {@link heatmap#tooltip})\n   * @param {tooltip} [_=none]\n   * @returns {heatmap | tooltip}\n   * @memberof heatmap\n   * @property\n   * by default tooltip = tooltip()\n   */\n  hm.tooltip = function(_) { return arguments.length ? (tooltip = _, hm) : tooltip; }\n\n  /**\n   * Gets / sets the colorFunction\n   * (see {@link heatmap#colorFunction})\n   * @param {colorFunction} [_=none]\n   * @returns {heatmap | colorFunction}\n   * @memberof heatmap\n   * @property\n   * by default colorFunction = colorFunction()\n   */\n  hm.colorFunction = function(_) { return arguments.length ? (colorFunction = _, hm) : colorFunction; }\n\n  /**\n   * Gets / sets the xSize\n   * (see {@link heatmap#xSize})\n   * @param {number} [_=none]\n   * @returns {heatmap | number}\n   * @memberof heatmap\n   * @property\n   * by default xSize = undefined\n   */\n  hm.xSize = function(_) { return arguments.length ? (xSize = _, hm) : xSize; }\n  /**\n   * Gets / sets the xSpacerSize\n   * (see {@link heatmap#xSpacerSize})\n   * @param {number} [_=none]\n   * @returns {heatmap | number}\n   * @memberof heatmap\n   * @property\n   * by default xSpacerSize = undefined\n   */\n  hm.xSpacerSize = function(_) { return arguments.length ? (xSpacerSize = _, hm) : xSpacerSize; }\n  /**\n   * Gets / sets the ySize\n   * (see {@link heatmap#ySize})\n   * @param {number} [_=none]\n   * @returns {heatmap | number}\n   * @memberof heatmap\n   * @property\n   * by default ySize = undefined\n   */\n  hm.ySize = function(_) { return arguments.length ? (ySize = _, hm) : ySize; }\n  /**\n   * Gets / sets the ySpacerSize\n   * (see {@link heatmap#ySpacerSize})\n   * @param {number} [_=none]\n   * @returns {heatmap | number}\n   * @memberof heatmap\n   * @property\n   * by default ySpacerSize = undefined\n   */\n  hm.ySpacerSize = function(_) { return arguments.length ? (ySpacerSize = _, hm) : ySpacerSize; }\n  // hm.yKeySortingFunction = function(_) { return arguments.length ? (yKeySortingFunction = _, hm) : yKeySortingFunction; }\n  // hm.xKeySortingFunction = function(_) { return arguments.length ? (xKeySortingFunction = _, hm) : xKeySortingFunction; }\n\n\n\n  function hm() {\n    var horizontalQ = true; // no orientation in heatmaps. Aids as placeholder in functions that take orient as a parameter\n    var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}\n    var container = setupContainer( selection, namespace, bgcpRect, backgroundFill );\n\n    cellKeys = d3.keys(data);\n\n    xValues = unique(cellKeys.map(xExtractor));\n    yValues = unique(cellKeys.map(yExtractor));\n    vValues = unique(cellKeys.map(vExtractor));\n\n    cellKeys.sort(function(a, b){ return xKeySortingFunction(a, b) || yKeySortingFunction(a, b) })\n    log('heatmap', 'cells are sorted by', cellKeys)\n\n\n\n    log('heatmap', 'x and y keys are', {x: xValues, y:yValues})\n\n\n    var xDim = xValues.length, yDim = yValues.length;\n\n\n    ySize = calculateWidthOfObject(spaceY, yDim, yMinObjectSize, yMaxObjectSize, objectSpacer, overflowQ)\n    xSize = calculateWidthOfObject(spaceX, xDim, xMinObjectSize, xMaxObjectSize, objectSpacer, overflowQ)\n    ySpacerSize = calculateWidthOfSpacer(yValues, spaceY, ySize, yDim, objectSpacer, overflowQ)\n    xSpacerSize = calculateWidthOfSpacer(xValues, spaceX, xSize, xDim, objectSpacer, overflowQ)\n    // console.table({\n    //     x:{\n    //       object: xSize,\n    //       spacer: xSpacerSize,\n    //       dim: xDim\n    //     },\n    //     y:{\n    //       object: ySize,\n    //       spacer: ySpacerSize,\n    //       dim: yDim\n    //     }\n    //\n    // })\n    log('heatmap', 'size of', {x: xSize, y: ySize})\n\n\n\n    var ySpacer = groupingSpacer()\n    .horizontalQ(false)\n    .moveby('category')\n    .numberOfObjects(yDim)\n    .objectClass(hypenate(objectClass, 'row'))\n    .objectSize(ySize + ySpacerSize)\n    .spacerSize(0)\n    .transitionDuration(transitionDuration)\n    .easeFunc(easeFunc)\n    .namespace('row')\n\n    var xSpacer = groupingSpacer()\n    .horizontalQ(true)\n    .moveby('category')\n    .numberOfObjects(xDim)\n    .objectClass(objectClass)\n    .objectSize(xSize + xSpacerSize)\n    .spacerSize(0)\n    .transitionDuration(transitionDuration)\n    .easeFunc(easeFunc)\n\n\n    ySpacer(container, yValues, 0)\n    container.selectAll('g.'+hypenate(objectClass, 'row'))\n    .each(function(d, i){ xSpacer(d3.select(this), xValues, 0) })\n\n    var cells = container.selectAll('g:not(.to-remove).'+objectClass)\n\n\n    if (cellKeys.length != yValues.length * xValues.length) {\n      var lookup = {}\n      cellKeys.map(function(k, i){\n        lookup[xExtractor(k)+'::'+yExtractor(k)] = k\n      })\n\n      var positionedCellKeys = []\n      for (var i = 0; i < yValues.length; i++) {\n        for (var j = 0; j < xValues.length; j++) {\n          var lookupValue = lookup[xValues[j]+\"::\"+yValues[i]]\n          if (lookupValue == undefined) {\n            positionedCellKeys.push(undefined)\n          } else {\n            positionedCellKeys.push(lookupValue)\n          }\n        }\n      }\n\n      cells.data(positionedCellKeys);\n\n      // maybe breaks this\n      // !!!!!! IMPORTANT NOTE TODO LOOK HERE BUG\n      cellKeys = positionedCellKeys\n    } else {\n      cells.data(cellKeys);\n    }\n\n\n\n    var parentIndexArray = []\n    cells.each(function(d, i){ parentIndexArray.push(Number(d3.select(this).attr('parent-index'))) })\n\n    colorFunction = colorFunction.colorBy() == 'index'\n    ? colorFunction.dataExtent([0, Math.max(...parentIndexArray)])\n    : colorFunction.dataExtent([0, Math.max(...vValues)])\n\n    cells.each(function(key, i) {\n      log('heatmap', 'each cell', {key: key, index: i, node: d3.select(this).node()})\n\n      var t = d3.select(this)\n      if (key == undefined) {return}\n      var currentData = data[key],\n      value = vExtractor(key, i),\n      i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'),\n      fillColor = colorFunction(key, value, i, 'fill'), // prevent duplicate computation\n      strokeColor = colorFunction(key, value, i,  'stroke')\n\n      var c = safeSelect(t, 'rect', hypenate(objectClass,'rect'))\n      c.attr('width', xSize + xSpacerSize - objectStrokeWidth)\n      .attr('height', ySize + ySpacerSize - objectStrokeWidth)\n      .attr('fill', fillColor)\n      .attr('x', objectStrokeWidth/2)\n      .attr('y', objectStrokeWidth/2)\n      .attr('stroke', \"#000\")\n      .attr('stroke-width', objectStrokeWidth)\n\n    })\n\n    tooltip.selection(cells.selectAll('rect.'+hypenate(objectClass, 'rect')))\n    .data(data)\n    // .keys(['r', 'v'])\n    // .header(function(d, i){return hypenate(data[d][xKey], data[d][yKey]) })\n\n    tooltip()\n\n\n  }\n\n  return hm;\n}\n\nexport {heatmap}\n","import {hypenate, safeSelect} from './helpers';\nimport {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer, whiskerPath} from './utils';\nimport {unique, hasQ, flatten} from './array-functions';\nimport {groupingSpacer} from './grouping-spacer';\nimport {colorFunction as CF} from './color-function';\nimport {tooltip as TTip} from './tooltip';\n/*******************************************************************************\n**                                                                            **\n**                                                                            **\n**                             BOX AND WHISKER                                **\n**                                                                            **\n**                                                                            **\n*******************************************************************************/\n/**\n * Creates a boxwhisker\n *\n * {@link https://sumneuron.gitlab.io/d3sm/demos/box-whiskers/index.html Demo}\n * @constructor boxwhisker\n * @param {d3.selection} selection\n * @namespace boxwhisker\n * @returns {function} boxwhisker\n */\nexport function boxwhisker( selection ) {\n  var\n  /**\n  * Data to plot. Assumed to be a object, where each key corresponds to a box\n  * (see {@link boxwhisker#data})\n  * @param {Object} [data=undefined]\n  * @memberof boxwhisker#\n  * @property\n  */\n  data,\n  /**\n  * Which direction to render the boxes in\n  * (see {@link boxwhisker#orient})\n  * @param {number} [orient='horizontal']\n  * @memberof boxwhisker#\n  * @property\n  */\n  orient = 'horizontal',\n  /**\n  * Amount of horizontal space (in pixels) avaible to render the boxwhisker in\n  * (see {@link boxwhisker#spaceX})\n  * @param {number} [spaceX=undefined]\n  * @memberof boxwhisker#\n  * @property\n  */\n  spaceX,\n  /**\n  * Amount of vertical space (in pixels) avaible to render the boxwhisker in\n  * (see {@link boxwhisker.spaceY})\n  * @param {number} [spaceY=undefined]\n  * @memberof boxwhisker#\n  * @property\n  */\n  spaceY,\n  /**\n  * Whether or not to allow boxwhisker to render elements pass the main spatial dimension\n  * given the orientation (see {@link boxwhisker#orient}), where {@link boxwhisker#orient}=\"horizontal\"\n  * the main dimension is {@link boxwhisker#spaceX} and where {@link boxwhisker#orient}=\"vertical\"\n  * the main dimension is {@link boxwhisker#spaceY}\n  * @param {boolean} [overflowQ=false]\n  * @memberof boxwhisker#\n  * @property\n  */\n  overflowQ = true,\n\n  /**\n  * An array - putatively of other arrays - depicting how boxes should be arranged\n  * @param {Array[]} [grouping=undefined]\n  * @memberof boxwhisker#\n  * @property\n  */\n  grouping,\n  quartilesKey = 'quartiles', // key in object where quartiles are stored\n  quartilesKeys = [\"0.00\", \"0.25\", \"0.50\", \"0.75\", \"1.00\"], // keys in quartiles object mapping values to min, q1, q2, q3 and max\n\n\n  /**\n  * How to get the value of the boxwhisker\n  * @param {function} [valueExtractor=function(key, index) { return data[key][quartilesKey] }]\n  * @memberof boxwhisker#\n  * @property\n  */\n  valueExtractor = function(key, index) { return data[key][quartilesKey] },\n  /**\n  * How to sort the boxes - if {@link boxwhisker#grouping} is not provided.\n  * @param {function} [sortingFunction=descending]\n  * @memberof boxwhisker#\n  * @property\n  * default\n  * function(keyA, keyB) {return d3.descending(\n  *   valueExtractor(keyA)[quartilesKeys[4]],\n  *   valueExtractor(keyB)[quartilesKeys[4]]\n  * )}\n  */\n  sortingFunction = function(keyA, keyB) {return d3.descending(\n    valueExtractor(keyA)[quartilesKeys[4]],\n    valueExtractor(keyB)[quartilesKeys[4]]\n  )},\n  /**\n  * The scale for which boxwhisker values should be transformed by\n  * @param {d3.scale} [scale=d3.scaleLinear]\n  * @memberof boxwhisker#\n  * @property\n  */\n  scale = d3.scaleLinear(),\n  /**\n  * The padding for the domain of the scale (see {@link boxwhisker#scale})\n  * @param {number} [domainPadding=0.5]\n  * @memberof boxwhisker#\n  * @property\n  */\n  domainPadding = 0.5,\n  /**\n  * Default space for the spacer (percentage) of main dimension given the orientation\n  * (see {@link boxwhisker#orient}), where {@link boxwhisker#orient}=\"horizontal\"\n  * the main dimension is {@link boxwhisker#spaceX} and where {@link boxwhisker#orient}=\"vertical\"\n  * the main dimension is {@link boxwhisker#spaceY} between boxes\n  * @param {number} [objectSpacer=0.05]\n  * @memberof boxwhisker#\n  * @property\n  */\n  objectSpacer = 0.05,\n  /**\n  * The minimum size that an object can be\n  * @param {number} [minObjectSize=15]\n  * @memberof boxwhisker#\n  * @property\n  */\n  minObjectSize = 15,\n  /**\n  * The maximum size that an object can be\n  * @param {number} [maxObjectSize=50]\n  * @memberof boxwhisker#\n  * @property\n  */\n  maxObjectSize = 50,\n  /**\n  * Percent of box and whisker size that whiskers will be rendered\n  * see {@link boxwhisker#objectSize}\n  * @param {number} [whiskerWidthPercent=0.6]\n  * @memberof boxwhisker#\n  * @property\n  */\n  whiskerWidthPercent = .6,\n  /**\n  * Instance of ColorFunction\n  * @param {function} [colorFunction = colorFunction()]\n  * @memberof boxwhisker#\n  * @property\n  */\n  colorFunction = CF(),\n  /**\n  * The stroke width of the boxes\n  * @param {number} [boxStrokeWidth=2]\n  * @memberof boxwhisker#\n  * @property\n  */\n  boxStrokeWidth = 2,\n  /**\n  * The stroke width of the whiskers\n  * @param {number} [whiskerStrokeWidth=2]\n  * @memberof boxwhisker#\n  * @property\n  */\n  whiskerStrokeWidth = 2,\n\n  /**\n  * Color of the background\n  * @param {string} [backgroundFill=\"transparent\"]\n  * @memberof boxwhisker#\n  * @property\n  */\n  backgroundFill = 'transparent',\n  /**\n  * Namespace for all items made by this instance of boxwhisker\n  * @param {string} [namespace=\"d3sm-box-whisker\"]\n  * @memberof boxwhisker#\n  * @property\n  */\n  namespace = 'd3sm-box-whisker',\n  /**\n  * Class name for boxwhisker container (<g> element)\n  * @param {string} [objectClass=\"box-whisk\"]\n  * @memberof boxwhisker#\n  * @property\n  */\n  objectClass = 'box-whisk',\n\n  /**\n  * Duration of all transitions of this element\n  * @param {number} [transitionDuration=1000]\n  * @memberof boxwhisker#\n  * @property\n  */\n  transitionDuration = 1000,\n  /**\n  * Easing function for transitions\n  * @param {d3.ease} [easeFunc=d3.easeExp]\n  * @memberof boxwhisker#\n  * @property\n  */\n  easeFunc = d3.easeExp,\n\n  /**\n  * The keys of the boxes\n  * @param {string[]} [boxKeys=undefined]\n  * @memberof boxwhisker#\n  * @property\n  */\n  boxKeys,\n  /**\n  * The values of the boxes\n  * @param {string[]} [boxValues=undefined]\n  * @memberof boxwhisker#\n  * @property\n  */\n  boxValues,\n  /**\n  * The objectSize (actual width) used by the boxes\n  * @param {number} [objectSize=undefined]\n  * @memberof boxwhisker#\n  * @property\n  */\n  objectSize,\n  /**\n  * The spacerSize (actual width) used by the spacers between the boxes\n  * @param {number} [spacerSize=undefined]\n  * @memberof boxwhisker#\n  * @property\n  */\n  spacerSize,\n  /**\n  * Instance of Tooltip\n  * @param {function} [tooltip=tooltip()]\n  * @memberof boxwhisker#\n  * @property\n  */\n  tooltip = TTip()\n  /**\n   * Gets or sets the selection in which items are manipulated\n   * @param {d3.selection} [_=none]\n   * @returns {boxwhisker | d3.selection}\n   * @memberof boxwhisker\n   * @property\n   * by default selection = selection\n   */\n  boxwhisker.selection = function(_) { return arguments.length ? (selection = _, boxwhisker) : selection; };\n  /**\n   * Gets or sets the data\n   * (see {@link boxwhisker#data})\n   * @param {number} [_=none]\n   * @returns {boxwhisker | object}\n   * @memberof boxwhisker\n   * @property\n   */\n  boxwhisker.data = function(_) { return arguments.length ? (data = _, boxwhisker) : data; };\n  /**\n   * Gets or sets the orient of the boxes\n   * (see {@link boxwhisker#orient})\n   * @param {number} [_=none]\n   * @returns {boxwhisker | object}\n   * @memberof boxwhisker\n   * @property\n   */\n  boxwhisker.orient = function(_) { return arguments.length ? (orient = _, boxwhisker) : orient; };\n  /**\n   * Gets or sets the amount of horizontal space in which items are manipulated\n   * (see {@link boxwhisker#spaceX})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {boxwhisker | number}\n   * @memberof boxwhisker\n   * @property\n   * by default spaceX = undefined\n   */\n  boxwhisker.spaceX = function(_) { return arguments.length ? (spaceX = _, boxwhisker) : spaceX; };\n  /**\n   * Gets or sets the amount of vertical space in which items are manipulated\n   * (see {@link boxwhisker#spaceY})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {boxwhisker | number}\n   * @memberof boxwhisker\n   * @property\n   * by default spaceY = undefined\n   */\n  boxwhisker.spaceY = function(_) { return arguments.length ? (spaceY = _, boxwhisker) : spaceY; };\n  /**\n   * Gets / sets whether or not boxwhisker is allowed to go beyond specified dimensions\n   * (see {@link boxwhisker#overflowQ})\n   * @param {boolean} [_=none]\n   * @returns {boxwhisker | boolean}\n   * @memberof boxwhisker\n   * @property\n   * by default overflowQ = false\n   */\n  boxwhisker.overflowQ = function(_) { return arguments.length ? (overflowQ = _, boxwhisker) : overflowQ; };\n  /**\n   * Gets / sets the grouping of the boxes\n   * (see {@link boxwhisker#grouping})\n   * @param {Array[]} [_=none]\n   * @returns {boxwhisker | Array[]}\n   * @memberof boxwhisker\n   * @property\n   * by default grouping = undefined\n   */\n  boxwhisker.grouping = function(_) { return arguments.length ? (grouping = _, boxwhisker) : grouping; };\n  /**\n   * Gets / sets the quartilesKey\n   * (see {@link boxwhisker#quartilesKey})\n   * @param {string} [_=none]\n   * @returns {boxwhisker | string}\n   * @memberof boxwhisker\n   * @property\n   * by default quartilesKey = quartiles\n   */\n  boxwhisker.quartilesKey = function(_) { return arguments.length ? (quartilesKey = _, boxwhisker) : quartilesKey; };\n  /**\n   * Gets / sets the quartilesKeys\n   * (see {@link boxwhisker#quartilesKeys})\n   * @param {string[]} [_=none]\n   * @returns {boxwhisker | string[]}\n   * @memberof boxwhisker\n   * @property\n   * by default quartilesKeys = [Q0, Q1, Q2, Q3, Q4]\n   */\n  boxwhisker.quartilesKeys = function(_) { return arguments.length ? (quartilesKeys = _, boxwhisker) : quartilesKeys; };\n  /**\n   * Gets / sets the valueExtractor\n   * (see {@link boxwhisker#valueExtractor})\n   * @param {function} [_=none]\n   * @returns {boxwhisker | function}\n   * @memberof boxwhisker\n   * @property\n   * by default valueExtractor = function(key, index) { return data[key][quartilesKey] },\n   */\n\n  boxwhisker.valueExtractor = function(_) { return arguments.length ? (valueExtractor = _, boxwhisker) : valueExtractor; };\n  /**\n   * Gets / sets the sortingFunction\n   * (see {@link boxwhisker#sortingFunction})\n   * @param {function} [_=none]\n   * @returns {boxwhisker | function}\n   * @memberof boxwhisker\n   * @property\n   * by default sortingFunction = function(keyA, keyB) {return d3.descending(\n   *   valueExtractor(keyA)[quartilesKeys[4]],\n   *   valueExtractor(keyB)[quartilesKeys[4]]\n   * )},\n   */\n  boxwhisker.sortingFunction = function(_) { return arguments.length ? (sortingFunction = _, boxwhisker) : sortingFunction; };\n  /**\n   * Gets / sets the scale for which the boxwhisker values should be transformed by\n   * (see {@link boxwhisker#scale})\n   * @param {d3.scale} [_=none]\n   * @returns {boxwhisker | d3.scale}\n   * @memberof boxwhisker\n   * @property\n   * by default scale = d3.scaleLinear()\n   */\n  boxwhisker.scale = function(_) { return arguments.length ? (scale = _, boxwhisker) : scale; };\n  /**\n   * Gets / sets the padding for the domain of the scale\n   * (see {@link boxwhisker#domainPadding})\n   * @param {number} [_=none]\n   * @returns {boxwhisker | number}\n   * @memberof boxwhisker\n   * @property\n   * by default domainPadding = 0.5\n   */\n  boxwhisker.domainPadding = function(_) { return arguments.length ? (domainPadding = _, boxwhisker) : domainPadding; };\n  /**\n   * Gets / sets objectSpacer\n   * (see {@link boxwhisker#objectSpacer})\n   * @param {number} [_=none]\n   * @returns {boxwhisker | number}\n   * @memberof boxwhisker\n   * @property\n   * by default objectSpacer = 0.05\n   */\n  boxwhisker.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, boxwhisker) : objectSpacer; };\n  /**\n   * Gets / sets the minObjectSize\n   * (see {@link boxwhisker#minObjectSize})\n   * @param {number} [_=none]\n   * @returns {boxwhisker | number}\n   * @memberof boxwhisker\n   * @property\n   * by default minObjectSize = 15\n   */\n  boxwhisker.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, boxwhisker) : minObjectSize; };\n  /**\n   * Gets / sets the maxObjectSize\n   * (see {@link boxwhisker#maxObjectSize})\n   * @param {number} [_=none]\n   * @returns {boxwhisker | number}\n   * @memberof boxwhisker\n   * @property\n   * by default maxObjectSize = 50\n   */\n  boxwhisker.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, boxwhisker) : maxObjectSize; };\n  /**\n   * Gets / sets the whiskerWidthPercent\n   * (see {@link boxwhisker#whiskerWidthPercent})\n   * @param {number} [_=none]\n   * @returns {boxwhisker | number}\n   * @memberof boxwhisker\n   * @property\n   * by default maxObjectSize = 0.6\n   */\n  boxwhisker.whiskerWidthPercent = function(_) { return arguments.length ? (whiskerWidthPercent = _, boxwhisker) : whiskerWidthPercent; };\n  /**\n   * Gets / sets the colorFunction\n   * (see {@link boxwhisker#colorFunction})\n   * @param {colorFunction} [_=none]\n   * @returns {boxwhisker | colorFunction}\n   * @memberof boxwhisker\n   * @property\n   * by default colorFunction = colorFunction()\n   */\n  boxwhisker.colorFunction = function(_) { return arguments.length ? (colorFunction = _, boxwhisker) : colorFunction; };\n  /**\n   * Gets / sets the boxStrokeWidth\n   * (see {@link boxwhisker#boxStrokeWidth})\n   * @param {number} [_=none]\n   * @returns {boxwhisker | number}\n   * @memberof boxwhisker\n   * @property\n   * by default boxStrokeWidth = 2\n   */\n  boxwhisker.boxStrokeWidth = function(_) { return arguments.length ? (boxStrokeWidth = _, boxwhisker) : boxStrokeWidth; };\n  /**\n   * Gets / sets the whiskerStrokeWidth\n   * (see {@link boxwhisker#whiskerStrokeWidth})\n   * @param {number} [_=none]\n   * @returns {boxwhisker | number}\n   * @memberof boxwhisker\n   * @property\n   * by default whiskerStrokeWidth = 2\n   */\n  boxwhisker.whiskerStrokeWidth = function(_) { return arguments.length ? (whiskerStrokeWidth = _, boxwhisker) : whiskerStrokeWidth; };\n\n  /**\n   * Gets / sets the backgroundFill\n   * (see {@link boxwhisker#backgroundFill})\n   * @param {string} [_=none]\n   * @returns {boxwhisker | string}\n   * @memberof boxwhisker\n   * @property\n   * by default backgroundFill = 'transparent'\n   */\n  boxwhisker.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, boxwhisker) : backgroundFill; };\n  /**\n   * Gets / sets the namespace\n   * (see {@link boxwhisker#namespace})\n   * @param {string} [_=none]\n   * @returns {boxwhisker | string}\n   * @memberof boxwhisker\n   * @property\n   * by default namespace = 'd3sm-boxwhisker'\n   */\n  boxwhisker.namespace = function(_) { return arguments.length ? (namespace = _, boxwhisker) : namespace; };\n  /**\n   * Gets / sets the objectClass\n   * (see {@link boxwhisker#objectClass})\n   * @param {string} [_=none]\n   * @returns {boxwhisker | string}\n   * @memberof boxwhisker\n   * @property\n   * by default objectClass = 'tick-group'\n   */\n  boxwhisker.objectClass = function(_) { return arguments.length ? (objectClass = _, boxwhisker) : objectClass; };\n  /**\n   * Gets / sets the transitionDuration\n   * (see {@link boxwhisker#transitionDuration})\n   * @param {number} [_=none]\n   * @returns {boxwhisker | number}\n   * @memberof boxwhisker\n   * @property\n   * by default transitionDuration = 1000\n   */\n  boxwhisker.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, boxwhisker) : transitionDuration; };\n  /**\n   * Gets / sets the easeFunc\n   * (see {@link boxwhisker#easeFunc})\n   * @param {d3.ease} [_=none]\n   * @returns {boxwhisker | d3.ease}\n   * @memberof boxwhisker\n   * @property\n   * by default easeFunc = d3.easeExp\n   */\n  boxwhisker.easeFunc = function(_) { return arguments.length ? (easeFunc = _, boxwhisker) : easeFunc; };\n\n  /**\n   * Gets / sets the barKeys\n   * (see {@link boxwhisker#boxKeys})\n   * @param {string[]} [_=none]\n   * @returns {boxwhisker | string[]}\n   * @memberof boxwhisker\n   * @property\n   * by default boxKeys = undefined\n   */\n  boxwhisker.boxKeys = function(_) { return arguments.length ? (boxKeys = _, boxwhisker) : boxKeys; };\n  /**\n   * Gets / sets the boxValues\n   * (see {@link boxwhisker#boxValues})\n   * @param {number[]} [_=none]\n   * @returns {boxwhisker | number[]}\n   * @memberof boxwhisker\n   * @property\n   * by default boxValues = undefined\n   */\n  boxwhisker.boxValues = function(_) { return arguments.length ? (boxValues = _, boxwhisker) : boxValues; };\n  /**\n   * Gets / sets the objectSize\n   * (see {@link boxwhisker#objectSize})\n   * @param {number} [_=none]\n   * @returns {boxwhisker | number}\n   * @memberof boxwhisker\n   * @property\n   * by default objectSize = undefined\n   */\n  boxwhisker.objectSize = function(_) { return arguments.length ? (objectSize = _, boxwhisker) : objectSize; };\n  /**\n   * Gets / sets the spacerSize\n   * (see {@link boxwhisker#spacerSize})\n   * @param {number} [_=none]\n   * @returns {boxwhisker | number}\n   * @memberof boxwhisker\n   * @property\n   * by default spacerSize = undefined\n   */\n  boxwhisker.spacerSize = function(_) { return arguments.length ? (spacerSize = _, boxwhisker) : spacerSize; };\n  /**\n   * Gets / sets the tooltip\n   * (see {@link boxwhisker#tooltip})\n   * @param {tooltip} [_=none]\n   * @returns {boxwhisker | tooltip}\n   * @memberof boxwhisker\n   * @property\n   * by default tooltip = tooltip()\n   */\n  boxwhisker.tooltip = function(_) { return arguments.length ? (tooltip = _, boxwhisker) : tooltip; };\n\n\n  function boxwhisker() {\n    // for convenience in handling orientation specific values\n    var horizontalQ = (orient == 'horizontal') ? true : false\n    var verticalQ = !horizontalQ\n\n    // background cliping rectangle\n    var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}\n    var container = setupContainer( selection, namespace, bgcpRect, backgroundFill );\n\n    // if grouping is undefined sort keys by sorting funct\n    var ordered = (grouping == undefined) ? d3.keys(data).sort(sortingFunction) : grouping\n    // to prevent re-calculation and getters to be passed to axes\n    boxKeys = flatten(ordered)\n    boxValues = boxKeys.map(valueExtractor)\n\n\n    var numberOfObjects = boxKeys.length\n    var extent = [\n      Math.min(...boxValues.map(function(d,i){return d[quartilesKeys[0]]})) - domainPadding,\n      Math.max(...boxValues.map(function(d,i){return d[quartilesKeys[4]]})) + domainPadding\n    ];\n\n    // set the scale\n    scale.domain(extent).range(horizontalQ ? [0,spaceY] : [spaceX, 0])\n    var space = horizontalQ ? spaceX : spaceY\n    // calculate object size\n    objectSize = calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ)\n    // calculate spacer size if needed\n    spacerSize = calculateWidthOfSpacer(boxKeys, space, objectSize, numberOfObjects, objectSpacer, overflowQ)\n    // make the nested groups\n    var spacerFunction = groupingSpacer()\n    .horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects)\n    .objectClass(objectClass).objectSize(objectSize).spacerSize(spacerSize)\n    .transitionDuration(transitionDuration).easeFunc(easeFunc)\n    .namespace(namespace)\n\n    // move stuff\n    spacerFunction(container, ordered, 0)\n\n    var parentIndexArray = []\n    container.selectAll('g:not(.to-remove).'+objectClass)\n    .each(function(d, i){if (hasQ(boxKeys, d)){ parentIndexArray.push(Number(d3.select(this).attr('parent-index')))}})\n\n\n\n    colorFunction = colorFunction.colorBy() == 'index'\n    ? colorFunction.dataExtent([0, Math.max(...parentIndexArray)])\n    : colorFunction.dataExtent(extent)\n\n\n    // set attributes for box and whiskers\n    container.selectAll('g:not(.to-remove).'+objectClass).each(function(key, i) {\n      var t = d3.select(this),\n      currentData = data[key],\n\n      quartiles = valueExtractor(key, i),\n      q0 = quartiles[quartilesKeys[0]],\n      q1 = quartiles[quartilesKeys[1]],\n      q2 = quartiles[quartilesKeys[2]],\n      q3 = quartiles[quartilesKeys[3]],\n      q4 = quartiles[quartilesKeys[4]]\n\n      var i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'),\n      fillColor = colorFunction(key, q2, i, 'fill'), // prevent duplicate computation\n      strokeColor = colorFunction(key, q2, i,  'stroke')\n\n\n      var\n      whisk = safeSelect(t, 'g', 'whisker'),\n      uWhisk = safeSelect(whisk, 'path', 'upper'),\n      lWhisk = safeSelect(whisk, 'path', 'lower'),\n      quart = safeSelect(t, 'g', 'quartile'),\n      uQuart = safeSelect(quart, 'rect', 'upper'),\n      lQuart = safeSelect(quart, 'rect', 'lower'),\n      mQuart = safeSelect(quart, 'circle', 'median')\n\n\n      // set upper quartile (q3)\n      uQuart.transition().duration(transitionDuration).ease(easeFunc)\n      .attr('width', horizontalQ ? objectSize : scale(q3) - scale(q2))\n      .attr('height', verticalQ ? objectSize : scale(q3) - scale(q2))\n      .attr('fill', fillColor)\n      .attr('stroke', strokeColor)\n      .attr('stroke-width', boxStrokeWidth)\n      .attr('transform', function(d, i){\n        var\n        x = horizontalQ ? 0 : scale(q2),\n        y = verticalQ ? 0 : scale(extent[1]) - scale(q3),\n        t = 'translate('+x+','+y+')'\n        return t\n      })\n\n      // set lower quartile (q1)\n      lQuart.transition().duration(transitionDuration).ease(easeFunc)\n      .attr('width', horizontalQ ? objectSize : scale(q2) - scale(q1))\n      .attr('height', verticalQ ? objectSize : scale(q2) - scale(q1))\n      .attr('fill', fillColor)\n      .attr('stroke', strokeColor)\n      .attr('stroke-width', boxStrokeWidth)\n      .attr('transform', function(d, i){\n        var\n        x = horizontalQ ? 0 : scale(q1),\n        y = verticalQ ? 0 : scale(extent[1]) - scale(q2),\n        t = 'translate('+x+','+y+')'\n        return t\n      })\n\n\n      // set median (q2)\n      mQuart.transition().duration(transitionDuration).ease(easeFunc)\n      .attr('r', function(d, i){\n        var r = objectSize / 2\n        var dif = (scale(q3) - scale(q1)) / 2\n        return (r > dif) ? dif : r\n      })\n      .attr('fill', fillColor)\n      .attr('stroke', strokeColor)\n      .attr('stroke-width', boxStrokeWidth)\n      .attr('transform', function(d, i){\n        var\n        x = horizontalQ ? objectSize / 2 : scale(q2),\n        y = verticalQ ? objectSize / 2 : scale(extent[1]) - scale(q2),\n        t = 'translate('+x+','+y+')'\n        return t\n      })\n\n      // set lower whisker (min)\n      lWhisk.transition().duration(transitionDuration).ease(easeFunc)\n      .attr('d', function(dd, ii){\n        var\n        dir = false,\n        x = 0,\n        y = 0,\n        h = horizontalQ ? scale(q1) - scale(q0) : objectSize,\n        w = verticalQ ? scale(q1) - scale(q0) : objectSize\n        return whiskerPath(dir, x, y, w, h, whiskerWidthPercent, orient)\n      })\n      .attr('transform', function(d, i){\n        var\n        x = horizontalQ ? 0 : scale(q1),\n        y = verticalQ ? 0 : scale(extent[1]) - scale(q1),\n        t = 'translate('+x+','+y+')'\n        return t\n      })\n      .attr('stroke', 'black').attr('stroke-width', whiskerStrokeWidth)\n      .attr('fill', 'none')\n\n      // set upper whisker (max)\n      uWhisk.transition().duration(transitionDuration).ease(easeFunc)\n      .attr('d', function(dd, ii){\n        var\n        dir = true,\n        x = 0,\n        y = 0,\n        h = horizontalQ ? scale(q4) - scale(q3) : objectSize,\n        w = verticalQ ? scale(q4) - scale(q3) : objectSize\n        return whiskerPath(dir, x, y, w, h, whiskerWidthPercent, orient)\n      })\n      .attr('transform', function(d, i){\n        var\n        x = horizontalQ ? 0 : scale(q3),\n        y = verticalQ ? 0 :  scale(extent[1]) - scale(q4),\n        t = 'translate('+x+','+y+')'\n        return t\n      })\n      .attr('stroke', 'black')\n      .attr('stroke-width', whiskerStrokeWidth)\n      .attr('fill', 'none')\n\n    })\n\n    tooltip.selection(container.selectAll('g:not(.to-remove).'+objectClass))\n    .data(data)\n    tooltip()\n\n\n  }\n\n  return boxwhisker\n}\n","import {hypenate, safeSelect} from './helpers';\nimport {selectFilter} from './select-filter';\n\n/*******************************************************************************\n**                                                                            **\n**                                                                            **\n**                                DATATOGGLE                                  **\n**                                                                            **\n**                                                                            **\n*******************************************************************************/\n/**\n * Creates a datatoggle\n * @constructor datatoggle\n * @param {d3.selection} selection\n * @namespace datatoggle\n * @returns {function} datatoggle\n */\nexport function datatoggle( selection ) {\n  var\n  /**\n  * Keys to make toggle-able options\n  * (see {@link datatoggle#keys})\n  * @param {string[]} [keys=undefined]\n  * @memberof datatoggle#\n  * @property\n  */\n  keys,\n  /**\n  * What to do when a different key is clicked\n  * (see {@link datatoggle#updateFunction})\n  * @param {function} [updateFunction=function(){}]\n  * @memberof datatoggle#\n  * @property\n  */\n  updateFunction = function(){},\n  /**\n  * Namespace for all items made by this instance of datatoggle\n  * @param {string} [namespace=\"d3sm-databar\"]\n  * @memberof datatoggle#\n  * @property\n  */\n  namespace='d3sm-databar',\n  /**\n  * Currently toggled key\n  * @param {string} [currentKey=undefined]\n  * @memberof datatoggle#\n  * @property\n  */\n  currentKey,\n\n  xAxisSelectQ = false,\n  xAxisOptions,\n  yAxisSelectQ = false,\n  yAxisOptions,\n  data\n  toggle.xAxisSelectQ = function(_){return arguments.length ? (xAxisSelectQ = _, toggle) : xAxisSelectQ}\n  toggle.yAxisSelectQ = function(_){return arguments.length ? (yAxisSelectQ = _, toggle) : yAxisSelectQ}\n  toggle.xAxisOptions = function(_){return arguments.length ? (xAxisOptions = _, toggle) : xAxisOptions}\n  toggle.yAxisOptions = function(_){return arguments.length ? (yAxisOptions = _, toggle) : yAxisOptions}\n  toggle.data = function(_){return arguments.length ? (data = _, toggle) : data}\n\n\n  /**\n   * Gets / sets the updateFunction\n   * (see {@link datatoggle#updateFunction})\n   * @function datatoggle.updateFunction\n   * @param {function} [_=none]\n   * @returns {datatoggle | function}\n   * @memberof datatoggle\n   * @property\n   * by default updateFunction = function(){}\n   */\n  toggle.updateFunction = function(_){return arguments.length ? (updateFunction = _, toggle) : updateFunction}\n  /**\n   * Gets / sets the namespace\n   * (see {@link datatoggle#namespace})\n   * @function datatoggle.namespace\n   * @param {string} [_=none]\n   * @returns {datatoggle | string}\n   * @memberof datatoggle\n   * @property\n   * by default namespace = 'd3sm-databar'\n   */\n  toggle.namespace = function(_){return arguments.length ? (namespace = _, toggle) : namespace}\n  /**\n   * Gets / sets the currentKey\n   * (see {@link datatoggle#currentKey})\n   * @function datatoggle.currentKey\n   * @param {string} [_=none]\n   * @returns {datatoggle | string}\n   * @memberof datatoggle\n   * @property\n   * by default currentKey = undefined\n   */\n  toggle.currentKeys =\n  function () {\n    var vals = {}\n    d3.keys(filterSelects).map(function(k, i){\n      vals[k]= filterSelects[k].currentOption()\n    })\n    return vals\n  }\n\n  var filterSelects = {}\n  function toggle() {\n    // selection options\n\n    // selection.classed('d-flex flex-row', true)\n    // var filterButton = safeSelect(selection, 'a', 'slider-buttons')\n    // var filterI = safeSelect(filterButton, 'i', 'fa fa-sliders')\n\n    /*BUG: unexpected behavior.\n      - when using bootstrap-eque way for collapse, clicking button submits to\n      the same page in applications (but not in demo), so using anchor (<a>)\n      - when using anchor, cause page to jump to that location\n      - when using show, the first open works, but the close does not.\n    */\n    // filterButton.attr('class', 'btn btn-secondary')\n    // .attr('data-toggle', 'collapse')\n    // .attr('href', '#'+hypenate(namespace, 'data-toggle'))\n    // .attr('target', '_blank')\n    // .html(filterButton.html()+' Filters')\n    // .on('click', function(d, i){\n    //   d3.event.preventDefault()\n    //   d3.event.stopPropagation()\n    //   var dt = d3.select(\"#\"+hypenate(namespace, 'data-toggle'))\n    //   dt.classed('show', !dt.classed('show'))\n    //   dt.classed('d-inline-flex', dt.classed('show'))\n    //\n    //   filterButton.classed('btn-primary', dt.classed('show'))\n    //   filterButton.classed('btn-secondary', !dt.classed('show'))\n    // })\n    // .classed('d-inline-flex', true)\n    // .style(\"margin-right\", '10px')\n\n\n\n    // var datatoggleCollapse = safeSelect(selection, 'div', hypenate(namespace,'collapse'))\n    // .attr('id', hypenate(namespace, 'data-toggle'))\n    // .classed('collapse', true)\n\n    // var flexRow = safeSelect(datatoggleCollapse, 'div', 'd-inline-flex flex-row flex-wrap')\n    var flexRow = safeSelect(selection, 'div', 'd-inline-flex flex-row flex-wrap')\n\n    var dataopts = flexRow.selectAll('div.'+hypenate(namespace,'select-filter'))\n    // remove excess\n    dataopts.exit().remove()\n    // bind data\n    dataopts = dataopts.data(d3.keys(data))\n    //enter\n    var doEnter = dataopts.enter().append('div')\n    .attr('class', 'select-filter')\n\n    dataopts = dataopts.merge(doEnter).style('margin-right', \"10px\")\n\n    dataopts.each(function(d, i){\n      var t = d3.select(this)\n      var sf = selectFilter(t)\n      .data(data[d])\n      .namespace(hypenate(namespace, d))\n      .selectionName(d)\n      sf()\n      filterSelects[d] = sf\n    })\n\n\n    selection.selectAll('select')\n    .on('change', function(){updateFunction()})\n    // bind update function\n    // d3.selectAll\n    return toggle\n  }\n\n\n  function onlyOne() {\n    // d3.event.preventDefault()\n    // d3.event.stopPropagation()\n    var dataopts = selection.selectAll('div.data-option')\n    currentKey = dataopts.select(':checked').datum()\n    updateFunction()\n\n  }\n\n  function axisSelectFilter(selection, axis=\"x-axis\", axisData) {\n    if (axisData == undefined) {\n      axisData = {\n        'linear': d3.scaleLinear(),\n        'log': d3.scaleLog(),\n        'pow': d3.scalePow(),\n        'sqrt': d3.scaleSqrt()\n      }\n    }\n\n    var sf = selectFilter(selection)\n    .data(axisData)\n    .namespace(axis)\n    .selectionName(axis+' scale')\n    .defaultValue(d3.scaleLinear())\n\n\n    sf()\n\n  }\n\n  return toggle\n}\n","import {hypenate, safeSelect} from './helpers';\nimport {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer} from './utils';\nimport {unique, flatten} from './array-functions';\nimport {groupingSpacer} from './grouping-spacer';\nimport {colorFunction as CF} from './color-function';\nimport {tooltip as TTip} from './tooltip';\n/*******************************************************************************\n**                                                                            **\n**                                                                            **\n**                                 SCATTER                                    **\n**                                                                            **\n**                                                                            **\n*******************************************************************************/\n/**\n * Creates a scatter\n *\n * {@link https://sumneuron.gitlab.io/d3sm/demos/scatter/index.html Demo}\n * @constructor scatter\n * @param {d3.selection} selection\n * @namespace scatter\n * @returns {function} scatter\n */\nexport function scatter ( selection ) {\n\n  var\n  /**\n  * Data to plot. Assumed to be a object, where each key corresponds to a point\n  * (see {@link scatter#data})\n  * @param {Object} [data=undefined]\n  * @memberof scatter#\n  * @property\n  */\n  data,\n  /**\n  * Amount of horizontal space (in pixels) avaible to render the scatter in\n  * (see {@link scatter#spaceX})\n  * @param {number} [spaceX=undefined]\n  * @memberof scatter#\n  * @property\n  */\n  spaceX,\n  /**\n  * Amount of vertical space (in pixels) avaible to render the scatter in\n  * (see {@link scatter.spaceY})\n  * @param {number} [spaceY=undefined]\n  * @memberof scatter#\n  * @property\n  */\n  spaceY,\n\n  /**\n  * The scale for which scatter x values should be transformed by\n  * @param {d3.scale} [scaleX=d3.scaleLinear]\n  * @memberof scatter#\n  * @property\n  */\n  scaleX = d3.scaleLinear(),\n  /**\n  * The padding for the domain of the scaleX (see {@link scatter#scaleX})\n  * @param {number} [domainPaddingX=0.5]\n  * @memberof scatter#\n  * @property\n  */\n  domainPaddingX = 0.5,\n  /**\n  * The function for getting the x value of the current point\n  * @param {function} [valueExtractorX=function(d, i){return data[d]['x']}]\n  * @memberof scatter#\n  * @property\n  */\n  valueExtractorX = function(d, i) {return data[d]['x']},\n\n  /**\n  * The scale for which scatter y values should be transformed by\n  * @param {d3.scale} [scaleY=d3.scaleLinear]\n  * @memberof scatter#\n  * @property\n  */\n  scaleY = d3.scaleLinear(),\n  /**\n  * The padding for the domain of the scaleY (see {@link scatter#scaleY})\n  * @param {number} [domainPaddingY=0.5]\n  * @memberof scatter#\n  * @property\n  */\n  domainPaddingY = 0.5,\n  /**\n  * The function for getting the y value of the current point\n  * @param {function} [valueExtractorY=function(d, i){return data[d]['y']}]\n  * @memberof scatter#\n  * @property\n  */\n  valueExtractorY = function(d, i) {return data[d]['y']},\n\n\n  /**\n  * The scale for which scatter r values should be transformed by\n  * @param {d3.scale} [scaleR=d3.scaleLinear]\n  * @memberof scatter#\n  * @property\n  */\n  scaleR = d3.scaleLinear(),\n  /**\n  * The padding for the domain of the scaleR (see {@link scatter#scaleR})\n  * @param {number} [domainPaddingR=0.5]\n  * @memberof scatter#\n  * @property\n  */\n  domainPaddingR = 0.5,\n  /**\n  * The function for getting the r value of the current point\n  * @param {function} [valueExtractorR=function(d, i){return 2}]\n  * @memberof scatter#\n  * @property\n  */\n  valueExtractorR = function(d, i) {return 2},\n  /**\n  * The min radius a point can have\n  * @param {function} [minRadius=2]\n  * @memberof scatter#\n  * @property\n  */\n  minRadius = 2,\n  /**\n  * The min radius a point can have\n  * @param {function} [maxRadius=10]\n  * @memberof scatter#\n  * @property\n  */\n  maxRadius = 10,\n\n  /**\n  * The stroke width of the points\n  * @param {number} [pointStrokeWidth=2]\n  * @memberof scatter#\n  * @property\n  */\n  pointStrokeWidth = 2,\n  /**\n  * Instance of ColorFunction\n  * @param {function} [colorFunction = colorFunction()]\n  * @memberof scatter#\n  * @property\n  */\n  colorFunction = CF(),\n  /**\n  * Color of the background\n  * @param {string} [backgroundFill=\"transparent\"]\n  * @memberof scatter#\n  * @property\n  */\n  backgroundFill = 'transparent',\n  /**\n  * Namespace for all items made by this instance of scatter\n  * @param {string} [namespace=\"d3sm-scatter\"]\n  * @memberof scatter#\n  * @property\n  */\n  namespace = 'd3sm-scatter',\n  /**\n  * Class name for scatter container (<circle> element)\n  * @param {string} [objectClass=\"scatter-point\"]\n  * @memberof scatter#\n  * @property\n  */\n  objectClass = 'scatter-point',\n  /**\n  * Duration of all transitions of this element\n  * @param {number} [transitionDuration=1000]\n  * @memberof scatter#\n  * @property\n  */\n  transitionDuration = 1000,\n  /**\n  * Easing function for transitions\n  * @param {d3.ease} [easeFunc=d3.easeExp]\n  * @memberof scatter#\n  * @property\n  */\n  easeFunc = d3.easeExp,\n\n  // useful values to extract to prevent re-calculation\n  /**\n  * The keys of the points\n  * @param {string[]} [pointKeys=undefined]\n  * @memberof scatter#\n  * @property\n  */\n  pointKeys,\n  /**\n  * The x values of the points\n  * @param {number[]} [valuesX=undefined]\n  * @memberof scatter#\n  * @property\n  */\n  valuesX,\n  /**\n  * The y values of the points\n  * @param {number[]} [valuesY=undefined]\n  * @memberof scatter#\n  * @property\n  */\n  valuesY,\n  /**\n  * The r values of the points\n  * @param {number[]} [valuesR=undefined]\n  * @memberof scatter#\n  * @property\n  */\n  valuesR,\n\n  /**\n  * Instance of Tooltip\n  * @param {function} [tooltip=tooltip()]\n  * @memberof scatter#\n  * @property\n  */\n  tooltip = TTip()\n\n  /**\n   * Gets or sets the selection in which items are manipulated\n   * @param {d3.selection} [_=none]\n   * @returns {scatter | d3.selection}\n   * @memberof scatter\n   * @property\n   * by default selection = selection\n   */\n  scatter.selection = function(_) { return arguments.length ? (selection =_, scatter) : selection}\n  /**\n   * Gets or sets the data\n   * (see {@link scatter#data})\n   * @param {number} [_=none]\n   * @returns {scatter | object}\n   * @memberof scatter\n   * @property\n   */\n  scatter.data = function(_) { return arguments.length ? (data =_, scatter) : data}\n  /**\n   * Gets or sets the amount of horizontal space in which items are manipulated\n   * (see {@link scatter#spaceX})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {scatter | number}\n   * @memberof scatter\n   * @property\n   * by default spaceX = undefined\n   */\n  scatter.spaceX = function(_) { return arguments.length ? (spaceX =_, scatter) : spaceX}\n  /**\n   * Gets or sets the amount of vertical space in which items are manipulated\n   * (see {@link scatter#spaceY})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {scatter | number}\n   * @memberof scatter\n   * @property\n   * by default spaceY = undefined\n   */\n  scatter.spaceY = function(_) { return arguments.length ? (spaceY =_, scatter) : spaceY}\n\n\n\n  /**\n   * Gets / sets the x scale for which the scatter x values should be transformed by\n   * (see {@link scatter#scaleX})\n   * @param {d3.scale} [_=none]\n   * @returns {scatter | d3.scale}\n   * @memberof scatter\n   * @property\n   * by default scaleX = d3.scaleLinear()\n   */\n  scatter.scaleX = function(_) { return arguments.length ? (scaleX =_, scatter) : scaleX}\n  /**\n   * Gets / sets the padding for the domain of the x scale\n   * (see {@link scatter#domainPaddingX})\n   * @param {number} [_=none]\n   * @returns {scatter | number}\n   * @memberof scatter\n   * @property\n   * by default domainPaddingX = 0.5\n   */\n  scatter.domainPaddingX = function(_) { return arguments.length ? (domainPaddingX =_, scatter) : domainPaddingX}\n  /**\n   * Gets / sets the valueExtractorX\n   * (see {@link scatter#valueExtractorX})\n   * @param {function} [_=none]\n   * @returns {scatter | function}\n   * @memberof scatter\n   * @property\n   * by default valueExtractorX = function(key, index) { return data[key]['x'] }\n   */\n  scatter.valueExtractorX = function(_) { return arguments.length ? (valueExtractorX =_, scatter) : valueExtractorX}\n\n\n  /**\n   * Gets / sets the y scale for which the scatter y values should be transformed by\n   * (see {@link scatter#scaleY})\n   * @param {d3.scale} [_=none]\n   * @returns {scatter | d3.scale}\n   * @memberof scatter\n   * @property\n   * by default scaleY = d3.scaleLinear()\n   */\n  scatter.scaleY = function(_) { return arguments.length ? (scaleY =_, scatter) : scaleY}\n  /**\n   * Gets / sets the padding for the domain of the y scale\n   * (see {@link scatter#domainPaddingY})\n   * @param {number} [_=none]\n   * @returns {scatter | number}\n   * @memberof scatter\n   * @property\n   * by default domainPaddingY = 0.5\n   */\n  scatter.domainPaddingY = function(_) { return arguments.length ? (domainPaddingY =_, scatter) : domainPaddingY}\n  /**\n   * Gets / sets the valueExtractorY\n   * (see {@link scatter#valueExtractorY})\n   * @param {function} [_=none]\n   * @returns {scatter | function}\n   * @memberof scatter\n   * @property\n   * by default valueExtractorY = function(key, index) { return data[key]['y'] }\n   */\n  scatter.valueExtractorY = function(_) { return arguments.length ? (valueExtractorY =_, scatter) : valueExtractorY}\n\n\n  /**\n   * Gets / sets the r scale for which the scatter r values should be transformed by\n   * (see {@link scatter#scaleR})\n   * @param {d3.scale} [_=none]\n   * @returns {scatter | d3.scale}\n   * @memberof scatter\n   * @property\n   * by default scaleR = d3.scaleLinear()\n   */\n  scatter.scaleR = function(_) { return arguments.length ? (scaleR =_, scatter) : scaleR}\n  /**\n   * Gets / sets the padding for the domain of the r scale\n   * (see {@link scatter#domainPaddingR})\n   * @param {number} [_=none]\n   * @returns {scatter | number}\n   * @memberof scatter\n   * @property\n   * by default domainPaddingR = 0.5\n   */\n  scatter.domainPaddingR = function(_) { return arguments.length ? (domainPaddingR =_, scatter) : domainPaddingR}\n  /**\n   * Gets / sets the valueExtractorR\n   * (see {@link scatter#valueExtractorR})\n   * @param {function} [_=none]\n   * @returns {scatter | function}\n   * @memberof scatter\n   * @property\n   * by default valueExtractorR = function(key, index) { return data[key]['r'] }\n   */\n  scatter.valueExtractorR = function(_) { return arguments.length ? (valueExtractorR =_, scatter) : valueExtractorR}\n  /**\n   * Gets / sets the minRadius\n   * (see {@link scatter#minRadius})\n   * @param {number} [_=none]\n   * @returns {scatter | number}\n   * @memberof scatter\n   * @property\n   * by default minRadius = 2\n   */\n  scatter.minRadius = function(_) { return arguments.length ? (minRadius =_, scatter) : minRadius}\n  /**\n   * Gets / sets the maxRadius\n   * (see {@link scatter#maxRadius})\n   * @param {number} [_=none]\n   * @returns {scatter | number}\n   * @memberof scatter\n   * @property\n   * by default maxRadius = 10\n   */\n  scatter.maxRadius = function(_) { return arguments.length ? (maxRadius =_, scatter) : maxRadius}\n\n  /**\n   * Gets / sets the pointStrokeWidth\n   * (see {@link scatter#pointStrokeWidth})\n   * @param {number} [_=none]\n   * @returns {scatter | number}\n   * @memberof scatter\n   * @property\n   * by default pointStrokeWidth = 2\n   */\n  scatter.pointStrokeWidth = function(_) { return arguments.length ? (pointStrokeWidth =_, scatter) : pointStrokeWidth}\n  /**\n   * Gets / sets the colorFunction\n   * (see {@link scatter#colorFunction})\n   * @param {number} [_=none]\n   * @returns {scatter | number}\n   * @memberof scatter\n   * @property\n   * by default colorFunction = colorFunction()\n   */\n  scatter.colorFunction = function(_) { return arguments.length ? (colorFunction =_, scatter) : colorFunction}\n  /**\n   * Gets / sets the backgroundFill\n   * (see {@link scatter#backgroundFill})\n   * @param {string} [_=none]\n   * @returns {scatter | string}\n   * @memberof scatter\n   * @property\n   * by default backgroundFill = 'transparent'\n   */\n  scatter.backgroundFill = function(_) { return arguments.length ? (backgroundFill =_, scatter) : backgroundFill}\n  /**\n   * Gets / sets the namespace\n   * (see {@link scatter#namespace})\n   * @param {string} [_=none]\n   * @returns {scatter | string}\n   * @memberof scatter\n   * @property\n   * by default namespace = 'd3sm-scatter'\n   */\n  scatter.namespace = function(_) { return arguments.length ? (namespace =_, scatter) : namespace}\n  /**\n   * Gets / sets the objectClass\n   * (see {@link scatter#objectClass})\n   * @param {string} [_=none]\n   * @returns {scatter | string}\n   * @memberof scatter\n   * @property\n   * by default objectClass = 'tick-group'\n   */\n  scatter.objectClass = function(_) { return arguments.length ? (objectClass =_, scatter) : objectClass}\n  /**\n   * Gets / sets the transitionDuration\n   * (see {@link scatter#transitionDuration})\n   * @param {number} [_=none]\n   * @returns {scatter | number}\n   * @memberof scatter\n   * @property\n   * by default transitionDuration = 1000\n   */\n  scatter.transitionDuration = function(_) { return arguments.length ? (transitionDuration =_, scatter) : transitionDuration}\n  /**\n   * Gets / sets the easeFunc\n   * (see {@link scatter#easeFunc})\n   * @param {d3.ease} [_=none]\n   * @returns {scatter | d3.ease}\n   * @memberof scatter\n   * @property\n   * by default easeFunc = d3.easeExp\n   */\n  scatter.easeFunc = function(_) { return arguments.length ? (easeFunc =_, scatter) : easeFunc}\n\n  /**\n   * Gets / sets the pointKeys\n   * (see {@link scatter#pointKeys})\n   * @param {string[]} [_=none]\n   * @returns {scatter | string[]}\n   * @memberof scatter\n   * @property\n   * by default pointKeys = undefined\n   */\n  scatter.pointKeys = function(_) { return arguments.length ? (pointKeys =_, scatter) : pointKeys}\n  /**\n   * Gets / sets the valuesX\n   * (see {@link scatter#valuesX})\n   * @param {number[]} [_=none]\n   * @returns {scatter | number[]}\n   * @memberof scatter\n   * @property\n   * by default valuesX = undefined\n   */\n  scatter.valuesX = function(_) { return arguments.length ? (valuesX =_, scatter) : valuesX}\n  /**\n   * Gets / sets the valuesY\n   * (see {@link scatter#valuesY})\n   * @param {number[]} [_=none]\n   * @returns {scatter | number[]}\n   * @memberof scatter\n   * @property\n   * by default valuesY = undefined\n   */\n  scatter.valuesY = function(_) { return arguments.length ? (valuesY =_, scatter) : valuesY}\n  /**\n   * Gets / sets the valuesR\n   * (see {@link scatter#valuesR})\n   * @param {number[]} [_=none]\n   * @returns {scatter | number[]}\n   * @memberof scatter\n   * @property\n   * by default valuesR = undefined\n   */\n  scatter.valuesR = function(_) { return arguments.length ? (valuesR =_, scatter) : valuesR}\n  /**\n   * Gets / sets the tooltip\n   * (see {@link scatter#tooltip})\n   * @param {tooltip} [_=none]\n   * @returns {scatter | tooltip}\n   * @memberof scatter\n   * @property\n   * by default tooltip = tooltip()\n   */\n\n  scatter.tooltip = function(_) { return arguments.length ? (tooltip =_, scatter) : tooltip}\n\n\n  function scatter() {\n    // background cliping rectangle\n    var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}\n    var container = setupContainer( selection, namespace, bgcpRect, backgroundFill );\n\n\n    pointKeys = d3.keys(data)\n    valuesX = pointKeys.map(valueExtractorX)\n    valuesY = pointKeys.map(valueExtractorY)\n    valuesR = pointKeys.map(valueExtractorR)\n\n    var numberOfObjects = pointKeys.length\n    var extentX = [Math.min(...valuesX) - domainPaddingX, Math.max(...valuesX) + domainPaddingX]\n    var extentY = [Math.min(...valuesY) - domainPaddingY, Math.max(...valuesY) + domainPaddingY]\n    var extentR = [Math.min(...valuesR) - domainPaddingR, Math.max(...valuesR) + domainPaddingR]\n\n    scaleX.domain(extentX).range([0, spaceX])\n    scaleY.domain(extentY).range([spaceY, 0])\n    scaleR.domain(extentR).range([minRadius, maxRadius])\n\n    var points = container.selectAll('.'+objectClass)\n    points = points.data(pointKeys)\n    var pEnter = points.enter().append('circle')\n    .attr('class', objectClass)\n    .attr('cx', 0).attr('cy', spaceY).attr('r', 0)\n\n    var pExit = points.exit()\n\n    points = points.merge(pEnter)\n\n    points.each(function(key, i){\n      var t = d3.select(this),\n      currentData = data[key],\n      x = valuesX[i],\n      y = valuesY[i],\n      r = valuesR[i],\n      fillColor = colorFunction(key, currentData, i, 'fill'),\n      strokeColor = colorFunction(key, currentData, i, 'stroke')\n\n      t.transition().duration(transitionDuration).ease(easeFunc)\n      .attr('cx', scaleX(x))\n      .attr('cy', scaleY(y))\n      .attr('r', scaleR(r))\n      .attr('fill', fillColor)\n      .attr('stroke', strokeColor)\n      .attr('stroke-width', pointStrokeWidth)\n\n\n\n      t.on('mouseover', function(d, i){\n        points.style('opacity', 0.2)\n        t.style('opacity', 1)\n        t.transition().duration(transitionDuration/2).ease(easeFunc)\n        .attr('stroke-width', pointStrokeWidth*2)\n        .attr('r', scaleR(r) * 1.5)\n\n      })\n      t.node().addEventListener('mouseout', function(){\n        container.selectAll('.'+objectClass).style('opacity', 1)\n        t.transition().duration(transitionDuration/2).ease(easeFunc)\n        .attr('stroke-width', pointStrokeWidth)\n        .attr('r', scaleR(r))\n\n      })\n\n    })\n\n\n\n    pExit.transition().duration(transitionDuration).ease(easeFunc)\n    .attr('cx', 0).attr('cy', spaceY).attr('r', 0)\n    .remove()\n\n    tooltip.selection(points)\n    .data(data)\n\n    tooltip()\n  }\n\n\n  return scatter\n}\n","import {hypenate, safeSelect, getTranslation} from './helpers';\nimport {log, warn, error, info} from './utils';\n/*******************************************************************************\n\n**                                                                            **\n**                                                                            **\n**                                PLOTZOOM                                    **\n**                                                                            **\n**                                                                            **\n*******************************************************************************/\n/**\n * Creates an plotZoom instance, which can handle drag and scroll events\n * @constructor plotZoom\n * @param {function} chart a function instance of one of the d3sm plots (e.g. bar, boxwhisker, bubbleHeatmap, violin, etc)\n * @param {axis}  xAxis the axis instance responsible for the x axis\n * @param {axis} yAxis the axis instance responsible for the y axis\n * @namespace plotZoom\n * @returns {function} zoom\n */\nexport function plotZoom( chart, xAxis, yAxis ) {\n  var\n  /**\n  * The event on which to fire\n  * (see {@link plotZoom#eventType})\n  * @param {string} eventType which event it should handle. Currently supports scroll and wheel\n  * @memberof plotZoom#\n  * @property\n  */\n  eventType,\n  /**\n  * A scaling factor for the wheel \"speed\"\n  * (see {@link plotZoom#wheelSpeed})\n  * @param {number} [wheelSpeed=20] scales the wheel translation by wheelSpeed\n  * @memberof plotZoom#\n  * @property\n  */\n  wheelSpeed = 20,\n  /**\n  * The orientation in which to allow scrolling: 'horizontal', 'vertical', or '2D'\n  * (see {@link plotZoom#orient})\n  * @param {string} [orient=chart.orient() || 'horizontal']\n  * @memberof plotZoom#\n  * @property\n  */\n  orient = (chart.orient == undefined) ? 'horizontal' : chart.orient(),\n  /**\n  * The max distance allowed to scroll in the x direction\n  * (see {@link plotZoom#xLock})\n  * @param {number} [xLock=chart.spaceX()] ideally chart.overflowQ() == true and this value is the\n  * bounding rect across all elements in the chart  minus the space in which to show.\n  * @memberof plotZoom#\n  * @property\n  */\n  xLock=chart.spaceX(),\n  /**\n  * The max distance allowed to scroll in the y direction\n  * (see {@link plotZoom#yLock})\n  * @param {number} [yLock=chart.spaceY()] ideally chart.overflowQ() == true and this value is the\n  * bounding rect across all elements in the chart  minus the space in which to show.\n  * @memberof plotZoom#\n  * @property\n  */\n  yLock=chart.spaceY(),\n\n  chartSel = chart.selection(),\n  xAxisSel = xAxis.selection(),\n  yAxisSel = yAxis.selection(),\n  svg = d3.select(chartSel.thisSVG())\n\n\n  /**\n   * Gets or sets the event type in which to respond\n   * (see {@link plotZoom#eventType})\n   * @param {string} [_=none] should be 'drag' or 'wheel'\n   * @returns {zoom | string}\n   * @memberof plotZoom\n   * @property\n   * by default plotZoom=undefined\n   */\n  zoom.eventType = function(_) { return arguments.length ? (eventType = _, zoom) : eventType; };\n  /**\n   * Gets or sets the wheel speed in which to scale the wheel scroll transform\n   * (see {@link plotZoom#wheelSpeed})\n   * @param {number} [_=none]\n   * @returns {zoom | number}\n   * @memberof plotZoom\n   * @property\n   * by default wheelSpeed=20\n   */\n  zoom.wheelSpeed = function(_) { return arguments.length ? (wheelSpeed = _, zoom) : wheelSpeed; };\n  /**\n   * Gets or sets the orientation in which items are manipulated\n   * (see {@link plotZoom#orient})\n   * @param {string} [_=none] should be horizontal, vertical, or 2D\n   * @returns {zoom | string}\n   * @memberof plotZoom\n   * @property\n   * by default orient=chart.orient() || 'horizontal'\n   */\n  zoom.orient = function(_) { return arguments.length ? (orient = _, zoom) : orient; };\n\n  /**\n   * Gets or sets the max distance in which to scroll X\n   * (see {@link plotZoom#xLock})\n   * @param {number} [_=none] should be a positive value\n   * @returns {zoom | number}\n   * @memberof plotZoom\n   * @property\n   * by default xLock=chart.spaceX()\n   */\n  zoom.xLock = function(_) { return arguments.length ? (xLock = _, zoom) : xLock; };\n  /**\n   * Gets or sets the max distance in which to scroll Y\n   * (see {@link plotZoom#yLock})\n   * @param {number} [_=none]  should be a positive value\n   * @returns {zoom | number}\n   * @memberof plotZoom\n   * @property\n   * by default yLock=chart.spaceY()\n   */\n  zoom.yLock = function(_) { return arguments.length ? (yLock = _, zoom) : yLock; };\n\n\n  function setLocks() {\n    var chartObjSel = chartSel.select('.'+hypenate(chart.namespace(),'object-container'))\n    var chartObjTrans = getTranslation(chartObjSel.attr('transform'))\n    var cos = chartObjSel.attr('transform', 'translate(0,0)')\n    xLock = chartSel.node().getBBox().width - chart.spaceX() * .9\n    yLock = chartSel.node().getBBox().height - chart.spaceY() * .9\n    cos.attr('transform', 'translate('+chartObjTrans[0]+','+chartObjTrans[1]+')')\n    log('plotZoom', 'setLocks', {xLock:xLock, yLock:yLock})\n  }\n\n\n  /**\n   * Sets the x and y locks (how far one can scroll)\n   * (see {@link plotZoom#xLock} and {@link plotZoom#yLock})\n   * @function plotZoom.setLocks\n   * @returns {undefined}\n   * @memberof plotZoom\n   * @property\n   */\n  zoom.setLocks = setLocks\n\n  function zoom() {\n    setLocks()\n\n    var horizontalQ, verticalQ\n    if (orient == '2D') {horizontalQ = true; verticalQ = true;}\n    if (orient == 'horizontal') {horizontalQ = true; verticalQ = false;}\n    if (orient == 'vertical') {verticalQ = true; horizontalQ = false;}\n\n    // capture transform event\n    var transform = d3.event.transform\n\n    var chartBox = chartSel.node().getBBox()\n    var xAxisBox = xAxisSel.node().getBBox()\n    var yAxisBox = xAxisSel.node().getBBox()\n\n    var chartWidth = chartBox.width - chartBox.x\n    var chartHeight = chartBox.height - chartBox.y\n    var xAxisWidth = xAxisBox.width// - xAxisBox.x\n    var xAxisHeight = xAxisBox.height// - xAxisBox.y\n    var yAxisWidth = yAxisBox.width// - yAxisBox.x\n    var yAxisHeight = yAxisBox.height// -yAxisBox.y\n\n    // enable wheel event\n    if (eventType == \"wheel\") {\n      var e = d3.event\n      // prevent page scrolling\n      e.preventDefault()\n      // event delta is very very slow, so speed it up with wheel speed\n      var w = d3.event.deltaY * wheelSpeed\n      var shiftQ = d3.event.shiftKey\n\n      // d3 has no way to make custom transform, so make an object and add\n      // the necessary functions to make wheel event compatible with drag events\n      if (orient == '2D') {\n        transform = shiftQ ? {k: 1, x: w, y: 0} : {k: 1, x: 0, y: w}\n      } else {\n        transform = horizontalQ ? {k: 1, x: w, y: 0} : {k: 1, x: 0, y: w}\n      }\n      // the * -1 inverts the direction\n      transform.applyX = function(x) { return x * this.k + this.x * -1; }\n      transform.applyY =  function(y) { return y * this.k + this.y * -1; }\n    }\n\n\n\n    var chartObjSel = chartSel.select('.'+hypenate(chart.namespace(),'object-container'))\n    var xAxisObjSel = xAxisSel.select('.'+hypenate(xAxis.namespace(),'object-container'))\n    var yAxisObjSel = yAxisSel.select('.'+hypenate(yAxis.namespace(),'object-container'))\n\n    // xLock = chartSel.node().getBBox().width - chart.spaceX() - chartSel.node().getBBox().x\n    // yLock = chartSel.node().getBBox().height - chart.spaceY()\n    // console.table({'xLock':xLock, \"yLock\":yLock})\n    // bhm.selection().node().getBBox().width - bhm.spaceX()\n\n\n    var chartObjTrans = getTranslation(chartObjSel.attr('transform'))\n    var xAxisObjTrans = getTranslation(xAxisObjSel.attr('transform'))\n    var yAxisObjTrans = getTranslation(yAxisObjSel.attr('transform'))\n\n\n    var x = horizontalQ ? transform.applyX(chartObjTrans[0]) : 0\n    if (horizontalQ) {x = x < -xLock ? (transform.x = 0, -xLock) : (transform.x = 0, Math.min(x, 0)) }\n\n    var y = verticalQ ? transform.applyY(chartObjTrans[1]) : 0\n    if (verticalQ) {y = y < -yLock ? (transform.y = 0, -yLock): (transform.y = 0, Math.min(y, 0))}\n\n    chartObjSel.attr('transform', 'translate('+x+','+y+')')\n    if (horizontalQ) { xAxisObjSel.attr('transform', 'translate('+x+','+0+')') }\n    if (verticalQ) { yAxisObjSel.attr('transform', 'translate('+0+','+y+')') }\n\n    // var lasso = svg.select(\".lasso-container\")\n    // if (!lasso.empty()) {\n    //   lasso.attr('transform', 'translate('+x+','+y+')')\n    // }\n\n  }\n\n  zoom.reset = function() {\n    var horizontalQ, verticalQ\n    if (orient == '2D') {horizontalQ = true; verticalQ = true;}\n    if (orient == 'horizontal') {horizontalQ = true; verticalQ = false;}\n    if (orient == 'vertical') {verticalQ = true; horizontalQ = false;}\n\n    var chartObjSel = chartSel.select('.'+hypenate(chart.namespace(),'object-container'))\n    var xAxisObjSel = xAxisSel.select('.'+hypenate(xAxis.namespace(),'object-container'))\n    var yAxisObjSel = yAxisSel.select('.'+hypenate(yAxis.namespace(),'object-container'))\n    chartObjSel.attr('transform', 'translate('+0+','+0+')')\n    xAxisObjSel.attr('transform', 'translate('+0+','+0+')')\n    yAxisObjSel.attr('transform', 'translate('+0+','+0+')')\n  }\n\n  return zoom\n}\n","import {hypenate, safeSelect, getTranslation} from './helpers';\nimport {log, warn, error, info} from './utils';\n/*******************************************************************************\n\n**                                                                            **\n**                                                                            **\n**                                PLOTZOOM                                    **\n**                                                                            **\n**                                                                            **\n*******************************************************************************/\n/**\n * Creates an plotZoom instance, which can handle drag and scroll events\n * @constructor plotZoom\n * @param {function} chart a function instance of one of the d3sm plots (e.g. bar, boxwhisker, bubbleHeatmap, violin, etc)\n * @param {axis}  xAxis the axis instance responsible for the x axis\n * @param {axis} yAxis the axis instance responsible for the y axis\n * @namespace plotZoom\n * @returns {function} zoom\n */\nexport function multiPlotZoom( chart ) {\n  var\n  /**\n  * The event on which to fire\n  * (see {@link plotZoom#eventType})\n  * @param {string} eventType which event it should handle. Currently supports scroll and wheel\n  * @memberof plotZoom#\n  * @property\n  */\n  eventType,\n  /**\n  * A scaling factor for the wheel \"speed\"\n  * (see {@link plotZoom#wheelSpeed})\n  * @param {number} [wheelSpeed=20] scales the wheel translation by wheelSpeed\n  * @memberof plotZoom#\n  * @property\n  */\n  wheelSpeed = 20,\n  /**\n  * The orientation in which to allow scrolling: 'horizontal', 'vertical', or '2D'\n  * (see {@link plotZoom#orient})\n  * @param {string} [orient=chart.orient() || 'horizontal']\n  * @memberof plotZoom#\n  * @property\n  */\n  orient = (chart.orient == undefined) ? 'horizontal' : chart.orient(),\n  /**\n  * The max distance allowed to scroll in the x direction\n  * (see {@link plotZoom#xLock})\n  * @param {number} [xLock=chart.spaceX()] ideally chart.overflowQ() == true and this value is the\n  * bounding rect across all elements in the chart  minus the space in which to show.\n  * @memberof plotZoom#\n  * @property\n  */\n  xLock=chart.spaceX(),\n  /**\n  * The max distance allowed to scroll in the y direction\n  * (see {@link plotZoom#yLock})\n  * @param {number} [yLock=chart.spaceY()] ideally chart.overflowQ() == true and this value is the\n  * bounding rect across all elements in the chart  minus the space in which to show.\n  * @memberof plotZoom#\n  * @property\n  */\n  yLock=chart.spaceY(),\n\n  chartSel = chart.selection(),\n\n  svg = d3.select(chartSel.thisSVG()),\n\n  xComponents = [],\n  yComponents = []\n\n\n  /**\n   * Gets or sets the event type in which to respond\n   * (see {@link plotZoom#eventType})\n   * @param {string} [_=none] should be 'drag' or 'wheel'\n   * @returns {zoom | string}\n   * @memberof plotZoom\n   * @property\n   * by default plotZoom=undefined\n   */\n  zoom.eventType = function(_) { return arguments.length ? (eventType = _, zoom) : eventType; };\n  /**\n   * Gets or sets the wheel speed in which to scale the wheel scroll transform\n   * (see {@link plotZoom#wheelSpeed})\n   * @param {number} [_=none]\n   * @returns {zoom | number}\n   * @memberof plotZoom\n   * @property\n   * by default wheelSpeed=20\n   */\n  zoom.wheelSpeed = function(_) { return arguments.length ? (wheelSpeed = _, zoom) : wheelSpeed; };\n  /**\n   * Gets or sets the orientation in which items are manipulated\n   * (see {@link plotZoom#orient})\n   * @param {string} [_=none] should be horizontal, vertical, or 2D\n   * @returns {zoom | string}\n   * @memberof plotZoom\n   * @property\n   * by default orient=chart.orient() || 'horizontal'\n   */\n  zoom.orient = function(_) { return arguments.length ? (orient = _, zoom) : orient; };\n\n  /**\n   * Gets or sets the max distance in which to scroll X\n   * (see {@link plotZoom#xLock})\n   * @param {number} [_=none] should be a positive value\n   * @returns {zoom | number}\n   * @memberof plotZoom\n   * @property\n   * by default xLock=chart.spaceX()\n   */\n  zoom.xLock = function(_) { return arguments.length ? (xLock = _, zoom) : xLock; };\n  /**\n   * Gets or sets the max distance in which to scroll Y\n   * (see {@link plotZoom#yLock})\n   * @param {number} [_=none]  should be a positive value\n   * @returns {zoom | number}\n   * @memberof plotZoom\n   * @property\n   * by default yLock=chart.spaceY()\n   */\n  zoom.yLock = function(_) { return arguments.length ? (yLock = _, zoom) : yLock; };\n\n  zoom.xComponents = function(_) { return arguments.length ? (xComponents = _, zoom) : xComponents; };\n  zoom.yComponents = function(_) { return arguments.length ? (yComponents = _, zoom) : yComponents; };\n\n\n  function setLocks() {\n    var chartObjSel = chartSel.select('.'+hypenate(chart.namespace(),'object-container'))\n    var chartObjTrans = getTranslation(chartObjSel.attr('transform'))\n    var cos = chartObjSel.attr('transform', 'translate(0,0)')\n\n    xLock = chartSel.node().getBBox().width - chart.spaceX()// * .9\n    yLock = chartSel.node().getBBox().height - chart.spaceY()// * .9\n    cos.attr('transform', 'translate('+chartObjTrans[0]+','+chartObjTrans[1]+')')\n    log('plotZoom', 'setLocks', {xLock:xLock, yLock:yLock})\n  }\n\n\n  /**\n   * Sets the x and y locks (how far one can scroll)\n   * (see {@link plotZoom#xLock} and {@link plotZoom#yLock})\n   * @function plotZoom.setLocks\n   * @returns {undefined}\n   * @memberof plotZoom\n   * @property\n   */\n  zoom.setLocks = setLocks\n\n  function zoom() {\n    setLocks()\n\n    var\n    xComponentsSel = xComponents.map(function(d, i){return d.selection()}),\n    yComponentsSel = yComponents.map(function(d, i){return d.selection()})\n\n    var horizontalQ, verticalQ\n    if (orient == '2D') {horizontalQ = true; verticalQ = true;}\n    if (orient == 'horizontal') {horizontalQ = true; verticalQ = false;}\n    if (orient == 'vertical') {verticalQ = true; horizontalQ = false;}\n\n    // capture transform event\n    var transform = d3.event.transform\n\n    var chartBox = chartSel.node().getBBox()\n    var xComponentsBox = xComponentsSel.map(function(d, i){return d.node().getBBox()})\n    var yComponentsBox = xComponentsSel.map(function(d, i){return d.node().getBBox()})\n\n\n    var chartWidth = chartBox.width - chartBox.x\n    var chartHeight = chartBox.height - chartBox.y\n\n    // enable wheel event\n    if (eventType == \"wheel\") {\n      var e = d3.event\n      // prevent page scrolling\n      e.preventDefault()\n      // event delta is very very slow, so speed it up with wheel speed\n      var w = d3.event.deltaY * wheelSpeed\n      var shiftQ = d3.event.shiftKey\n\n      // d3 has no way to make custom transform, so make an object and add\n      // the necessary functions to make wheel event compatible with drag events\n      if (orient == '2D') {\n        transform = shiftQ ? {k: 1, x: w, y: 0} : {k: 1, x: 0, y: w}\n      } else {\n        transform = horizontalQ ? {k: 1, x: w, y: 0} : {k: 1, x: 0, y: w}\n      }\n      // * -1 is invert\n      transform.applyX = function(x) { return x * this.k + this.x *-1; }\n      transform.applyY =  function(y) { return y * this.k + this.y *-1; }\n    }\n\n\n\n    var chartObjSel = chartSel.select('.'+hypenate(chart.namespace(),'object-container'))\n    var xComponentObjSel = xComponentsSel.map(function(d, i){\n      return d.select('.'+hypenate(xComponents[i].namespace(),'object-container'))\n    })\n    var yComponentObjSel = yComponentsSel.map(function(d, i){\n      return d.select('.'+hypenate(yComponents[i].namespace(),'object-container'))\n    })\n\n    var chartObjTrans = getTranslation(chartObjSel.attr('transform'))\n    var xComponentsObjTrans = xComponentObjSel.map(function(d, i){\n      return getTranslation(d.attr('transform'))\n    })\n    var yComponentsObjTrans = yComponentObjSel.map(function(d, i){\n      return getTranslation(d.attr('transform'))\n    })\n\n    var x = horizontalQ ? transform.applyX(chartObjTrans[0]) : 0\n    if (horizontalQ) {x = x < -xLock ? (transform.x = 0, -xLock) : (transform.x = 0, Math.min(x, 0)) }\n\n    var y = verticalQ ? transform.applyY(chartObjTrans[1]) : 0\n    if (verticalQ) {y = y < -yLock ? (transform.y = 0, -yLock): (transform.y = 0, Math.min(y, 0))}\n\n    chartObjSel.attr('transform', 'translate('+x+','+y+')')\n    if (horizontalQ) {\n      // xAxisObjSel.attr('transform', 'translate('+x+','+0+')')\n      xComponentObjSel.map(function(d, i){ d.attr('transform', 'translate('+x+','+0+')')  })\n    }\n    if (verticalQ) {\n      // yAxisObjSel.attr('transform', 'translate('+0+','+y+')')\n      yComponentObjSel.map(function(d, i){ d.attr('transform', 'translate('+0+','+y+')')  })\n\n    }\n\n\n  }\n\n  zoom.reset = function() {\n    var horizontalQ, verticalQ\n    if (orient == '2D') {horizontalQ = true; verticalQ = true;}\n    if (orient == 'horizontal') {horizontalQ = true; verticalQ = false;}\n    if (orient == 'vertical') {verticalQ = true; horizontalQ = false;}\n\n    var chartObjSel = chartSel.select('.'+hypenate(chart.namespace(),'object-container'))\n    var xAxisObjSel = xAxisSel.select('.'+hypenate(xAxis.namespace(),'object-container'))\n    var yAxisObjSel = yAxisSel.select('.'+hypenate(yAxis.namespace(),'object-container'))\n    chartObjSel.attr('transform', 'translate('+0+','+0+')')\n    xAxisObjSel.attr('transform', 'translate('+0+','+0+')')\n    yAxisObjSel.attr('transform', 'translate('+0+','+0+')')\n  }\n\n  return zoom\n}\n","import {hypenate, safeSelect, modifyHexidecimalColorLuminance, extractViolinValues, quartiles} from './helpers';\nimport {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer} from './utils';\nimport {unique, hasQ, flatten, whichBin} from './array-functions';\nimport {groupingSpacer} from './grouping-spacer';\nimport {colorFunction as CF} from './color-function';\nimport {tooltip as TTip} from './tooltip';\n\n/*******************************************************************************\n**                                                                            **\n**                                                                            **\n**                                 VIOLIN                                     **\n**                                                                            **\n**                                                                            **\n*******************************************************************************/\n\n/**\n * Creates a violin\n *\n * {@link https://sumneuron.gitlab.io/d3sm/demos/basic-violins/index.html Demo}\n * @constructor violin\n * @param {d3.selection} selection\n * @namespace violin\n * @returns {function} violin\n */\nexport function violin( selection ) {\n  var\n  /**\n  * Data to plot. Assumed to be a object, where each key corresponds to a violin\n  * (see {@link violin#data})\n  * @param {Object} [data=undefined]\n  * @memberof violin#\n  * @property\n  */\n  data,\n  /**\n  * Which direction to render the bars in\n  * (see {@link violin#orient})\n  * @param {number} [orient='horizontal']\n  * @memberof violin#\n  * @property\n  */\n  orient='horizontal',\n  /**\n  * Amount of horizontal space (in pixels) avaible to render the violin in\n  * (see {@link violin#spaceX})\n  * @param {number} [spaceX=undefined]\n  * @memberof violin#\n  * @property\n  */\n  spaceX,\n  /**\n  * Amount of vertical space (in pixels) avaible to render the violin in\n  * (see {@link violin.spaceY})\n  * @param {number} [spaceY=undefined]\n  * @memberof violin#\n  * @property\n  */\n  spaceY,\n  /**\n  * Whether or not to allow violin to render elements pass the main spatial dimension\n  * given the orientation (see {@link violin#orient}), where {@link violin#orient}=\"horizontal\"\n  * the main dimension is {@link violin#spaceX} and where {@link violin#orient}=\"vertical\"\n  * the main dimension is {@link violin#spaceY}\n  * @param {boolean} [overflowQ=false]\n  * @memberof violin#\n  * @property\n  */\n  overflowQ = true,\n  /**\n  * Whether or not to display points inside the points\n  * @param {boolean} [pointsQ=false]\n  * @memberof violin#\n  * @property\n  */\n  pointsQ = true,\n  /**\n  * An array - putatively of other arrays - depicting how bars should be arranged\n  * @param {Array[]} [grouping=undefined]\n  * @memberof violin#\n  * @property\n  */\n  grouping,\n  /**\n  * How to get the value of the violin\n  * @param {function} [valueExtractor=function(key, index) { return data[key] }]\n  * @memberof violin#\n  * @property\n  */\n  valueExtractor = function(key, index) {return data[key] },\n  /**\n  * How to sort the bars - if {@link violin#grouping} is not provided.\n  * @param {function} [sortingFunction=function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}]\n  * @memberof violin#\n  * @property\n  */\n  sortingFunction = function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])},\n\n  /**\n  * The scale for which violin values should be transformed by\n  * @param {d3.scale} [scale=d3.scaleLinear]\n  * @memberof violin#\n  * @property\n  */\n  scale = d3.scaleLinear(),\n  /**\n  * The padding for the domain of the scale (see {@link violin#scale})\n  * @param {number} [domainPadding=0.5]\n  * @memberof violin#\n  * @property\n  */\n  domainPadding = 0.5,\n  /**\n  * Default space for the spacer (percentage) of main dimension given the orientation\n  * (see {@link violin#orient}), where {@link violin#orient}=\"horizontal\"\n  * the main dimension is {@link violin#spaceX} and where {@link violin#orient}=\"vertical\"\n  * the main dimension is {@link violin#spaceY} between bars\n  * @param {number} [objectSpacer=0.05]\n  * @memberof violin#\n  * @property\n  */\n  objectSpacer = 0.05,\n  /**\n  * The minimum size that an object can be\n  * @param {number} [minObjectSize=50]\n  * @memberof violin#\n  * @property\n  */\n  minObjectSize = 50,\n  /**\n  * The maximum size that an object can be\n  * @param {number} [maxObjectSize=100]\n  * @memberof violin#\n  * @property\n  */\n  maxObjectSize = 100,\n\n  /**\n  * The stroke width of the bars\n  * @param {number} [barStrokeWidth=2]\n  * @memberof violin#\n  * @property\n  */\n  objectStrokeWidth = 2,\n  /**\n  * Instance of ColorFunction\n  * @param {function} [colorFunction = colorFunction()]\n  * @memberof violin#\n  * @property\n  */\n  colorFunction = CF(),\n  /**\n  * Instance of ColorFunction modified by a scale for the points\n  * @param {function} [pointColorFunc = colorFunction()]\n  * @memberof violin#\n  * @property\n  */\n  pointColorFunc = function (d, type, base, min, max) {\n    var minMaxHexScale = d3.scaleLinear().domain([min, max]).range([-0.25, 0.05])\n    var scaledColor = modifyHexidecimalColorLuminance(base.replace('#', ''), minMaxHexScale(d))\n    var mod = type == \"stroke\" ? 0 : 0.25\n    return modifyHexidecimalColorLuminance(scaledColor.replace('#', ''), mod)\n  },\n\n  /**\n  * The radius of a point\n  * @param {number} [pointRadius=3]\n  * @memberof violin#\n  * @property\n  */\n  pointRadius = 3,\n  /**\n  * The stroke width of the oints\n  * @param {number} [pointStrokeWidth=2]\n  * @memberof violin#\n  * @property\n  */\n  pointStrokeWidth = 2,\n\n  /**\n  * Color of the background\n  * @param {string} [backgroundFill=\"transparent\"]\n  * @memberof violin#\n  * @property\n  */\n  backgroundFill = 'transparent',\n  /**\n  * Namespace for all items made by this instance of violin\n  * @param {string} [namespace=\"d3sm-violin\"]\n  * @memberof violin#\n  * @property\n  */\n  namespace = 'd3sm-violin',\n  /**\n  * Class name for violin container (<g> element)\n  * @param {string} [objectClass=\"violin\"]\n  * @memberof violin#\n  * @property\n  */\n  objectClass = 'violin',\n  /**\n  * Duration of all transitions of this element\n  * @param {number} [transitionDuration=1000]\n  * @memberof violin#\n  * @property\n  */\n  transitionDuration = 1000,\n  /**\n  * Easing function for transitions\n  * @param {d3.ease} [easeFunc=d3.easeExp]\n  * @memberof violin#\n  * @property\n  */\n  easeFunc = d3.easeExp,\n\n  /**\n  * The key containing the quartiles\n  * @param {string} [quartilesKey=undefined]\n  * @memberof violin#\n  * @property\n  */\n  quartilesKey = \"quartiles\",\n  /**\n  * The keys corresponding to each quartile\n  * @param {string[]} [quartileKeys=[\"Q0\", \"Q1\", \"Q2\", \"Q3\", \"Q4\"]]\n  * @memberof violin#\n  * @property\n  */\n  quartileKeys = [\"Q0\", \"Q1\", \"Q2\", \"Q3\", \"Q4\"],\n\n  /**\n  * The keys of the bars\n  * @param {string[]} [violinKeys=undefined]\n  * @memberof violin#\n  * @property\n  */\n  violinKeys,\n  /**\n  * The values of the bars\n  * @param {number[]} [violinValues=undefined]\n  * @memberof violin#\n  * @property\n  */\n  violinValues,\n  /**\n  * The objectSize (actual width) used by the bars\n  * @param {number} [objectSize=undefined]\n  * @memberof violin#\n  * @property\n  */\n  objectSize,\n  /**\n  * The spacerSize (actual width) used by the spacers between the bars\n  * @param {number} [spacerSize=undefined]\n  * @memberof violin#\n  * @property\n  */\n  spacerSize,\n\n  /**\n  * Instance of Tooltip\n  * @param {function} [tooltip=tooltip()]\n  * @memberof violin#\n  * @property\n  */\n  tooltip = TTip().keys([quartileKeys[4], quartileKeys[3], quartileKeys[2], quartileKeys[1], quartileKeys[0]]),\n  pointsTooltip = TTip(),\n  // pointKeyExtractor = function(violinKey, violinData, violinValues) {return d3.keys(violinValues[violinKey].values)},\n  // pointValueExtractor = function(pointKey, violinKey, violinData, violinValues) {return violinValues[violinKey].values[pointKey]},\n\n\n  /**\n  * Function which given the key of the violin and that key's associated value\n  * returns the object consiting of the points of the violin\n  * (see {@link violin#violinPointsExtractor})\n  * @param {Object} [violinPointsExtractor=function(violinKey, violinData) { return violinData.points }]\n  * @memberof violin#\n  * @property\n  */\n  violinPointsExtractor = function (violinKey, violinData) {return violinData.points },\n  /**\n  * Function which given the key of the current point and the object of points for the\n  * violin, returns the numerical value of the point\n  * (see {@link violin#violinPointValueExtractor})\n  * @param {Object} [violinPointValueExtractor=function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value }]\n  * @memberof violin#\n  * @property\n  */\n  violinPointValueExtractor = function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value }\n\n\n  /**\n   * Gets or sets the violinPointsExtractor\n   * @param {function} [_=none]\n   * @returns {violin | function}\n   * @memberof violin\n   * @property\n   * by default violinPointsExtractor = function(violinKey, violinData) { return violinData.points }\n   */\n  violin.violinPointsExtractor = function(_) { return arguments.length ? (violinPointsExtractor = _, violin) : violinPointsExtractor; };\n  /**\n   * Gets or sets the violinPointValueExtractor\n   * @param {function} [_=none]\n   * @returns {violin | function}\n   * @memberof violin\n   * @property\n   * by default violinPointsExtractor = function(pointKey, violinKey, violinData, violinValues) {return violinValues[violinKey].values[pointKey]}\n   */\n  violin.violinPointValueExtractor = function(_) { return arguments.length ? (violinPointValueExtractor = _, violin) : violinPointValueExtractor; };\n\n\n  /**\n   * Gets or sets the selection in which items are manipulated\n   * @param {d3.selection} [_=none]\n   * @returns {violin | d3.selection}\n   * @memberof violin\n   * @property\n   * by default selection = selection\n   */\n  violin.selection = function(_) { return arguments.length ? (selection = _, violin) : selection; };\n  /**\n   * Gets or sets the data\n   * (see {@link violin#data})\n   * @param {number} [_=none]\n   * @returns {violin | object}\n   * @memberof violin\n   * @property\n   */\n  violin.data = function(_) { return arguments.length ? (data = _, violin) : data; };\n  /**\n   * Gets or sets the orient of the boxes\n   * (see {@link violin#orient})\n   * @param {number} [_=none]\n   * @returns {violin | object}\n   * @memberof violin\n   * @property\n   */\n  violin.orient = function(_) { return arguments.length ? (orient = _, violin) : orient; };\n  /**\n   * Gets or sets the amount of horizontal space in which items are manipulated\n   * (see {@link violin#spaceX})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {violin | number}\n   * @memberof violin\n   * @property\n   * by default spaceX = undefined\n   */\n  violin.spaceX = function(_) { return arguments.length ? (spaceX = _, violin) : spaceX; };\n  /**\n   * Gets or sets the amount of vertical space in which items are manipulated\n   * (see {@link violin#spaceY})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {violin | number}\n   * @memberof violin\n   * @property\n   * by default spaceY = undefined\n   */\n  violin.spaceY = function(_) { return arguments.length ? (spaceY = _, violin) : spaceY; };\n\n\n  /**\n   * Gets / sets whether or not violin is allowed to go beyond specified dimensions\n   * (see {@link violin#overflowQ})\n   * @param {boolean} [_=none]\n   * @returns {violin | boolean}\n   * @memberof violin\n   * @property\n   * by default overflowQ = false\n   */\n  violin.overflowQ = function(_) { return arguments.length ? (overflowQ = _, violin) : overflowQ; };\n  /**\n   * Gets / sets whether or not to plot points with the violins\n   * (see {@link violin#pointsQ})\n   * @param {boolean} [_=none]\n   * @returns {violin | boolean}\n   * @memberof violin\n   * @property\n   * by default pointsQ = false\n   */\n  violin.pointsQ = function(_) { return arguments.length ? (pointsQ = _, violin) : pointsQ; };\n\n\n  /**\n   * Gets / sets the grouping of the boxes\n   * (see {@link violin#grouping})\n   * @param {Array[]} [_=none]\n   * @returns {violin | Array[]}\n   * @memberof violin\n   * @property\n   * by default grouping = undefined\n   */\n  violin.grouping = function(_) { return arguments.length ? (grouping = _, violin) : grouping; };\n  /**\n   * Gets / sets the valueExtractor\n   * (see {@link violin#valueExtractor})\n   * @param {function} [_=none]\n   * @returns {violin | function}\n   * @memberof violin\n   * @property\n   */\n  violin.valueExtractor = function(_) { return arguments.length ? (valueExtractor = _, violin) : valueExtractor; };\n  /**\n   * Gets / sets the sortingFunction\n   * (see {@link violin#sortingFunction})\n   * @param {function} [_=none]\n   * @returns {violin | function}\n   * @memberof violin\n   * @property\n   */\n  violin.sortingFunction = function(_) { return arguments.length ? (sortingFunction = _, violin) : sortingFunction; };\n\n  /**\n   * Gets / sets the scale for which the violin values should be transformed by\n   * (see {@link violin#scale})\n   * @param {d3.scale} [_=none]\n   * @returns {violin | d3.scale}\n   * @memberof violin\n   * @property\n   * by default scale = d3.scaleLinear()\n   */\n  violin.scale = function(_) { return arguments.length ? (scale = _, violin) : scale; };\n  /**\n   * Gets / sets the padding for the domain of the scale\n   * (see {@link violin#domainPadding})\n   * @param {number} [_=none]\n   * @returns {violin | number}\n   * @memberof violin\n   * @property\n   * by default domainPadding = 0.5\n   */\n  violin.domainPadding = function(_) { return arguments.length ? (domainPadding = _, violin) : domainPadding; };\n\n\n  /**\n   * Gets / sets objectSpacer\n   * (see {@link violin#objectSpacer})\n   * @param {number} [_=none]\n   * @returns {violin | number}\n   * @memberof violin\n   * @property\n   * by default objectSpacer = 0.05\n   */\n  violin.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, violin) : objectSpacer; };\n  /**\n   * Gets / sets the minObjectSize\n   * (see {@link violin#minObjectSize})\n   * @param {number} [_=none]\n   * @returns {violin | number}\n   * @memberof violin\n   * @property\n   * by default minObjectSize = 15\n   */\n  violin.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, violin) : minObjectSize; };\n  /**\n   * Gets / sets the maxObjectSize\n   * (see {@link violin#maxObjectSize})\n   * @param {number} [_=none]\n   * @returns {violin | number}\n   * @memberof violin\n   * @property\n   * by default maxObjectSize = 50\n   */\n  violin.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, violin) : maxObjectSize; };\n\n  /**\n   * Gets / sets the objectStrokeWidth\n   * (see {@link violin#objectStrokeWidth})\n   * @param {number} [_=none]\n   * @returns {violin | number}\n   * @memberof violin\n   * @property\n   * by default objectStrokeWidth = 2\n   */\n  violin.objectStrokeWidth = function(_) { return arguments.length ? (objectStrokeWidth = _, violin) : objectStrokeWidth; };\n\n\n  /**\n   * Gets / sets the colorFunction\n   * (see {@link violin#colorFunction})\n   * @param {colorFunction} [_=none]\n   * @returns {violin | colorFunction}\n   * @memberof violin\n   * @property\n   * by default colorFunction = colorFunction()\n   */\n  violin.colorFunction = function(_) { return arguments.length ? (colorFunction = _, violin) : colorFunction; };\n  /**\n   * Gets / sets the colorFunction\n   * (see {@link violin#colorFunction})\n   * @param {colorFunction} [_=none]\n   * @returns {violin | colorFunction}\n   * @memberof violin\n   * @property\n   * by default colorFunction = colorFunction()\n   */\n  violin.pointColorFunc = function(_) { return arguments.length ? (pointColorFunc = _, violin) : pointColorFunc; };\n\n\n  /**\n   * Gets / sets the pointRadius\n   * (see {@link violin#pointRadius})\n   * @param {number} [_=none]\n   * @returns {violin | number}\n   * @memberof violin\n   * @property\n   * by default pointRadius = 2\n   */\n  violin.pointRadius = function(_) { return arguments.length ? (pointRadius = _, violin) : pointRadius; };\n  /**\n   * Gets / sets the pointStrokeWidth\n   * (see {@link violin#pointStrokeWidth})\n   * @param {number} [_=none]\n   * @returns {violin | number}\n   * @memberof violin\n   * @property\n   * by default pointStrokeWidth = 2\n   */\n  violin.pointStrokeWidth = function(_) { return arguments.length ? (pointStrokeWidth = _, violin) : pointStrokeWidth; };\n\n\n  /**\n   * Gets / sets the backgroundFill\n   * (see {@link violin#backgroundFill})\n   * @param {string} [_=none]\n   * @returns {violin | string}\n   * @memberof violin\n   * @property\n   * by default backgroundFill = 'transparent'\n   */\n  violin.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, violin) : backgroundFill; };\n  /**\n   * Gets / sets the namespace\n   * (see {@link violin#namespace})\n   * @param {string} [_=none]\n   * @returns {violin | string}\n   * @memberof violin\n   * @property\n   * by default namespace = 'd3sm-violin'\n   */\n  violin.namespace = function(_) { return arguments.length ? (namespace = _, violin) : namespace; };\n  /**\n   * Gets / sets the objectClass\n   * (see {@link violin#objectClass})\n   * @param {string} [_=none]\n   * @returns {violin | string}\n   * @memberof violin\n   * @property\n   * by default objectClass = 'tick-group'\n   */\n  violin.objectClass = function(_) { return arguments.length ? (objectClass = _, violin) : objectClass; };\n\n\n  /**\n   * Gets / sets the transitionDuration\n   * (see {@link violin#transitionDuration})\n   * @param {number} [_=none]\n   * @returns {violin | number}\n   * @memberof violin\n   * @property\n   * by default transitionDuration = 1000\n   */\n  violin.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, violin) : transitionDuration; };\n  /**\n   * Gets / sets the easeFunc\n   * (see {@link violin#easeFunc})\n   * @param {d3.ease} [_=none]\n   * @returns {violin | d3.ease}\n   * @memberof violin\n   * @property\n   * by default easeFunc = d3.easeExp\n   */\n  violin.easeFunc = function(_) { return arguments.length ? (easeFunc = _, violin) : easeFunc; };\n\n\n  /**\n   * Gets / sets the quartileKey\n   * (see {@link violin#quartileKey})\n   * @param {string} [_=none]\n   * @returns {violin | string}\n   * @memberof violin\n   * @property\n   * by default quartileKey = \"quartiles\"\n   */\n  violin.quartileKey = function(_) { return arguments.length ? (quartileKey = _, violin) : quartileKey; };\n  /**\n   * Gets / sets the quartileKeys\n   * (see {@link violin#quartileKeys})\n   * @param {string[]} [_=none]\n   * @returns {violin | string[]}\n   * @memberof violin\n   * @property\n   * by default quartileKeys = [\"Q0\",\"Q1\",\"Q2\",\"Q3\",\"Q4\"]\n   */\n  violin.quartileKeys = function(_) { return arguments.length ? (quartileKeys = _, violin) : quartileKeys; };\n\n\n  /**\n   * Gets / sets the violinKeys\n   * (see {@link violin#violinKeys})\n   * @param {string[]} [_=none]\n   * @returns {violin | string[]}\n   * @memberof violin\n   * @property\n   * by default violinKeys = undefined\n   */\n  violin.violinKeys = function(_) { return arguments.length ? (violinKeys = _, violin) : violinKeys; };\n  /**\n   * Gets / sets the violinValues\n   * (see {@link violin#violinValues})\n   * @param {Object[]} [_=none]\n   * @returns {violin | Object[]}\n   * @memberof violin\n   * @property\n   * by default violinValues = undefined\n   */\n  violin.violinValues = function(_) { return arguments.length ? (violinValues = _, violin) : violinValues; };\n\n  /**\n   * Gets / sets the objectSize\n   * (see {@link violin#objectSize})\n   * @param {number} [_=none]\n   * @returns {violin | number}\n   * @memberof violin\n   * @property\n   * by default objectSize = undefined\n   */\n  violin.objectSize = function(_) { return arguments.length ? (objectSize = _, violin) : objectSize; };\n  /**\n   * Gets / sets the spacerSize\n   * (see {@link violin#spacerSize})\n   * @param {number} [_=none]\n   * @returns {violin | number}\n   * @memberof violin\n   * @property\n   * by default spacerSize = undefined\n   */\n  violin.spacerSize = function(_) { return arguments.length ? (spacerSize = _, violin) : spacerSize; };\n  /**\n   * Gets / sets the tooltip\n   * (see {@link violin#tooltip})\n   * @param {tooltip} [_=none]\n   * @returns {violin | tooltip}\n   * @memberof violin\n   * @property\n   * by default tooltip = tooltip()\n   */\n  violin.tooltip = function(_) { return arguments.length ? (tooltip = _, violin) : tooltip; };\n  /**\n   * Gets / sets the pointsTooltip\n   * (see {@link violin#pointsTooltip})\n   * @param {tooltip} [_=none]\n   * @returns {violin | tooltip}\n   * @memberof violin\n   * @property\n   * by default pointsTooltip = tooltip()\n   */\n  violin.pointsTooltip = function(_) { return arguments.length ? (pointsTooltip = _, violin) : pointsTooltip; };\n\n\n\n\n\n  function violin () {\n    // for convenience in handling orientation specific values\n    var horizontalQ = (orient == 'horizontal') ? true : false\n    var verticalQ = !horizontalQ\n\n    // background cliping rectangle\n    var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}\n    var container = setupContainer( selection, namespace, bgcpRect, backgroundFill );\n\n    // if grouping is undefined sort violinKeys by sortingFunction\n    var ordered = (grouping == undefined) ? d3.keys(data).sort(sortingFunction) : grouping\n\n    // console.log(ordered)\n\n    violinKeys = flatten(ordered)\n\n\n    var calcValues = neededViolinValues()\n    .horizontalQ(horizontalQ)\n    .quartileKeys(quartileKeys)\n    .violinPointsExtractor(violinPointsExtractor)\n    .violinPointValueExtractor(violinPointValueExtractor)\n\n\n\n    // augment valus\n    violinKeys.map(function(vk, i){ calcValues(vk, data) })\n\n    var numberOfObjects = violinKeys.length\n\n    var min = [].concat(...violinKeys.map(function(k, i){return data[k].quartiles[quartileKeys[0]]}))\n    var max = [].concat(...violinKeys.map(function(k, i){return data[k].quartiles[quartileKeys[quartileKeys.length - 1]]}))\n    var extent = [Math.min(...min) - domainPadding, Math.max(...max) + domainPadding]\n    // console.log(extent, violinValues, ordered)\n\n    // set the scale\n    scale.domain(extent).range(horizontalQ ? [0,spaceY] : [0, spaceX])\n    var space = horizontalQ ? spaceX : spaceY\n    // calculate object size\n    objectSize = calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ)\n    // calculate spacer size if needed\n    spacerSize = calculateWidthOfSpacer(ordered, space, objectSize, numberOfObjects, objectSpacer, overflowQ)\n    // make the nested groups\n    var spacerFunction = groupingSpacer()\n    .horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects)\n    .objectClass(objectClass).objectSize(objectSize).spacerSize(spacerSize)\n    .transitionDuration(transitionDuration).easeFunc(easeFunc)\n    .namespace(namespace)\n\n    // move stuff\n    spacerFunction(container, ordered, 0)\n    // console.log(violinKeys, ordered, container.selectAll('g:not(.to-remove).'+objectClass).nodes())\n\n    // for color function\n    var parentIndexArray = []\n    container.selectAll('g:not(.to-remove).'+objectClass)\n    .each(function(d, i){if (hasQ(violinKeys, d)){ parentIndexArray.push(Number(d3.select(this).attr('parent-index')))}})\n\n    // update color function\n    colorFunction = colorFunction.colorBy() == 'index'\n    ? colorFunction.dataExtent([0, Math.max(...parentIndexArray)])\n    : colorFunction.dataExtent(extent)\n\n    /* violiin specific needs */\n\n\n    var frequencyMax = Math.max(...[].concat(...violinKeys.map(function(k, i){return d3.max(data[k].frequencies)})))\n    var vScale = d3.scaleLinear().domain([0, frequencyMax]).range([0, objectSize / 2])\n\n    var lArea = d3.line()\n    .x(function(d, i){ return horizontalQ ? -vScale(d.x) : scale(d.x)})\n    .y(function(d, i){ return horizontalQ ? scale(extent[1]) - scale(d.y) : -vScale(d.y)})\n    .curve(d3.curveBasis)\n    var rArea = d3.line()\n    .x(function(d, i){ return horizontalQ ? vScale(d.x) : scale(d.x)})\n    .y(function(d, i){ return horizontalQ ? scale(extent[1]) - scale(d.y) : vScale(d.y)})\n    .curve(d3.curveBasis)\n\n\n\n\n\n\n    container.selectAll('g:not(.to-remove).'+objectClass).each(function(key, i){\n      var t = d3.select(this),\n      currentData = data[key]\n      // needed because bug in selecting .to-remove\n      if (!hasQ(violinKeys, key)) {return}\n      var\n      i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'),\n      fillColor = colorFunction(key, currentData, i, 'fill'), // prevent duplicate computation\n      strokeColor = colorFunction(key, currentData, i, 'stroke'),\n      area = safeSelect(t, 'g', 'area'),\n      la = safeSelect(area, 'path', 'left'),\n      ra = safeSelect(area, 'path', 'right'),\n      quarts = safeSelect(t, 'g', 'quarts'),\n      lq3 = safeSelect(quarts, 'line', 'q3'),\n      lq1 = safeSelect(quarts, 'line', 'q1'),\n      q3 = currentData.quartiles[quartileKeys[3]],\n      q2 = currentData.quartiles[quartileKeys[2]],\n      q1 = currentData.quartiles[quartileKeys[1]]\n\n      t.attr('transform', horizontalQ ? 'translate('+objectSize / 2+',0)' : 'translate(0,'+objectSize / 2+')'  )\n      // draw curve\n      la.transition().duration(transitionDuration).attr('d', function(dd, ii){ return lArea(currentData.contour)})\n      .attr('fill', fillColor)\n      .attr('stroke', strokeColor)\n      .attr('stroke-width', objectStrokeWidth)\n\n      ra.transition().duration(transitionDuration).attr('d', function(dd, ii){ return rArea(currentData.contour)})\n      .attr('fill', fillColor)\n      .attr('stroke', strokeColor)\n      .attr('stroke-width', objectStrokeWidth)\n\n      area.node().addEventListener('mouseover', function(dd, ii){\n        container.selectAll('g.'+objectClass).style('opacity', 0.2)\n        t.style('opacity', 1)\n        la.attr('stroke-width',objectStrokeWidth*2)\n        ra.attr('stroke-width',objectStrokeWidth*2)\n      })\n      area.node().addEventListener('mouseout', function(dd, ii){\n        container.selectAll('g.'+objectClass).style('opacity', 1)\n        la.attr('stroke-width',objectStrokeWidth)\n        ra.attr('stroke-width',objectStrokeWidth)\n      })\n\n      if (pointsQ) {\n        var ptsContainer = safeSelect(t, 'g', 'points')\n        var pts = ptsContainer.selectAll('.point').data(currentData.pointKeys)\n        pts.on('mouseover', null)\n\n\n        var ptsExit = pts.exit().transition().ease(easeFunc).duration(transitionDuration)\n        .attr('r', 0)\n        .attr('cy', horizontalQ ? scale(extent[1]) - scale(q2) : vScale(0))\n        .attr('cx', horizontalQ ? vScale(0) : scale(q2)).remove()\n\n        var ptsEnter = pts.enter().append('circle').attr('class', 'point').attr('r', 0)\n        .attr('cx', horizontalQ ? 0 : scale(q2))\n        .attr('cy', horizontalQ ? scale(q2) : 0)\n\n        pts = pts.merge(ptsEnter)\n\n        // console.log(pointsTooltip.header())\n\n        var pTTips = TTip().selection(pts).data(violinPointsExtractor(key, currentData))\n        .header(pointsTooltip.header())\n        .keys(pointsTooltip.keys())\n        .values(pointsTooltip.values())\n\n        pTTips()\n\n        var pMin = d3.min(currentData.pointValues), pMax = d3.max(currentData.pointValues)\n\n\n\n        pts.transition().duration(transitionDuration).ease(easeFunc).attr('r', pointRadius)\n        .attr('cy', function(pointKey, ii){\n          var dd = currentData.pointValues[ii]\n          if (horizontalQ) { return scale(extent[1]) - scale(dd) }\n          var j = whichBin(currentData.binned, dd)\n          var r = Math.random()\n          var n = vScale(r * currentData.frequencies[j] * 0.5)\n          var k = Math.random() > 0.5 ? n : -n\n          return k\n        })\n        .attr('cx', function(pointKey, ii){\n          var dd = currentData.pointValues[ii]\n          if (horizontalQ) {\n            var j = whichBin(currentData.binned, dd)\n            var r = Math.random()\n            var n = vScale(r * currentData.frequencies[j] * 0.5)\n            var k = Math.random() > 0.5 ? n : -n\n            return k\n          }\n          return scale(dd)\n        })\n        .attr('stroke', function(dd, ii) { var dd = currentData.pointValues[ii]; return pointColorFunc(dd, 'stroke', strokeColor, pMin, pMax) })\n        .attr('fill'  , function(dd, ii) { var dd = currentData.pointValues[ii]; return pointColorFunc(dd, 'fill'  , strokeColor, pMin, pMax) })\n        .attr('stroke-width', pointStrokeWidth)\n\n        ptsContainer.selectAll('circle.point').on('mouseover', function(dd, ii){\n          container.selectAll('g.'+objectClass).style('opacity', 0.2)\n          t.style('opacity', 1)\n          la.attr('stroke-width',objectStrokeWidth*2)\n          ra.attr('stroke-width',objectStrokeWidth*2)\n\n          container.selectAll('.point').style('opacity', 0.2)\n          d3.select(this).style('opacity', 1).attr('r', pointRadius * 2).attr('stroke-width',pointStrokeWidth*2)\n        })\n        ptsContainer.selectAll('circle.point').on('mouseout', function(dd, ii){\n          var e = document.createEvent('SVGEvents')\n          e.initEvent('mouseout',true,true);\n          area.node().dispatchEvent(e)\n\n          container.selectAll('.point').style('opacity', 1)\n          d3.select(this).attr('stroke-width', pointStrokeWidth).attr('r', pointRadius)\n        })\n      }\n      else {\n        cV.selectAll('.point')\n        .transition().duration(transitionDuration).ease(easeFunc)\n        .attr('r', 0)\n        .attr('cy', horizontalQ ? scale(extent[1]) - scale(q2) : vScale(0))\n        .attr('cx', horizontalQ ? vScale(0) : scale(q2))\n        .remove()\n      }\n\n\n    })\n\n\n    tooltip.selection(container.selectAll('g:not(.to-remove).'+objectClass + ' .area'))\n    if (tooltip.data() == undefined) {tooltip.data(data)}\n    tooltip()\n    if (tooltip.values() == undefined) {\n      tooltip.values([\n        function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] },\n        function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] },\n        function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] },\n        function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] },\n        function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] }\n      ])\n\n    }\n\n  }\n\n  return violin\n}\n\n\n\n\n\n/**\n* Produces the function which manipulates the violin data to have values needed\n* for rendering the violins as svg.\n* @returns {function} calculateViolinValues\n* @namespace neededViolinValues\n*/\nfunction neededViolinValues() {\n  var\n  /**\n  * Whether or not the orientation of the violins are horizontal\n  * (see {@link violin#orient})\n  * @param {Object} [horizontalQ=true]\n  * @memberof neededViolinValues#\n  * @property\n  */\n  horizontalQ = true,\n  /**\n  * Keys to be put into the quartiles if they need to be calculated.\n  * (see {@link violin#quartileKeys})\n  * @param {Object} [quartileKeys=['Q0', 'Q1', 'Q2', 'Q3', 'Q4']]\n  * @memberof neededViolinValues#\n  * @property\n  */\n  quartileKeys = ['Q0', 'Q1', 'Q2', 'Q3', 'Q4'],\n  /**\n  * Function which given the key of the violin and that key's associated value\n  * returns the object consiting of the points of the violin\n  * (see {@link violin#violinPointsExtractor})\n  * @param {Object} [violinPointsExtractor=function(violinKey, violinData) { return violinData.points }]\n  * @memberof neededViolinValues#\n  * @property\n  */\n  violinPointsExtractor,\n  /**\n  * Function which given the key of the current point and the object of points for the\n  * violin, returns the numerical value of the point\n  * (see {@link violin#violinPointValueExtractor})\n  * @param {Object} [violinPointValueExtractor=function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value }]\n  * @memberof neededViolinValues#\n  * @property\n  */\n  violinPointValueExtractor\n\n\n  /**\n   * Gets / sets the horizontalQ\n   * (see {@link violin#orient})\n   * @param {boolean} [_=none]\n   * @returns {calculateViolinValues | boolean}\n   * @memberof calculateViolinValues\n   * @property\n   *\n   * by default horizontalQ = true\n   */\n  calculateViolinValues.horizontalQ = function(_) { return arguments.length ? (horizontalQ=_, calculateViolinValues) : horizontalQ }\n  /**\n   * Gets / sets the quartileKeys\n   * (see {@link violin#quartileKeys})\n   * @param {string[]} [_=none]\n   * @returns {calculateViolinValues | string[]}\n   * @memberof calculateViolinValues\n   * @property\n   *\n   * by default quartileKeys = [\"Q0\",\"Q1\",\"Q2\",\"Q3\",\"Q4\"]\n   */\n  calculateViolinValues.quartileKeys = function(_) { return arguments.length ? (quartileKeys=_, calculateViolinValues) : quartileKeys }\n  /**\n   * Gets / sets the violinPointsExtractor\n   * (see {@link violin#violinPointsExtractor})\n   * @param {function} [_=none]\n   * @returns {calculateViolinValues | function}\n   * @memberof calculateViolinValues\n   * @property\n   *\n   * by default violinPointsExtractor = function(violinKey, violinData) { return violinData.points }\n   */\n  calculateViolinValues.violinPointsExtractor = function(_) { return arguments.length ? (violinPointsExtractor=_, calculateViolinValues) : violinPointsExtractor }\n  /**\n   * Gets / sets the violinPointValueExtractor\n   * (see {@link violin#violinPointValueExtractor})\n   * @param {function} [_=none]\n   * @returns {calculateViolinValues | function}\n   * @memberof calculateViolinValues\n   * @property\n   *\n   * by default violinPointValueExtractor = function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value }\n   */\n  calculateViolinValues.violinPointValueExtractor = function(_) { return arguments.length ? (violinPointValueExtractor=_, calculateViolinValues) : violinPointValueExtractor }\n\n\n  /**\n  * The function produced by neededViolinValues.\n  *\n  * Adds the data need to render the violin as an svg\n  * @param {string} violinKey the key of the current violin\n  * @param {object} data the object consisting of violinKey - values (violin data) pairs\n  * @returns {none} this function manipulates the passed data object adding the following keys\n  *\n  * data[violinKey].binned // the binned values of the points\n  *\n  * data[violinKey].frequencies // the list consisting of the length of each bin\n  *\n  * data[violinKey].contour // the points depicting the contour of 1/2 of the violin\n  *\n  * data[violinKey].quartiles // the quartiles of the points' values\n  *\n  * data[violinKey].pointKeys // the keys associated with the points\n  *\n  * data[violinKey].pointValues // the numerical values of the points\n  *\n  * @memberof neededViolinValues#\n  * @property\n  */\n  function calculateViolinValues(violinKey, data) {\n    // data for the current violin\n    var violinData = data[violinKey];\n    // the object of points\n    var violinPoints = violinPointsExtractor(violinKey, violinData);\n    //\n    var violinPointsKeys = d3.keys(violinPoints);\n    // the numerical values of those points\n    var violinPointsValues = violinPointsKeys.map(function(pk, i){return violinPointValueExtractor(pk, violinPoints)})\n\n    // quartiles of those points\n    var pointQuartiles = quartiles(violinPointsValues, quartileKeys)\n\n    // binned points\n    var binned = d3.histogram()(violinPointsValues)\n    // length of bins\n    var frequencies = binned.map(bin=>bin.length)\n    // min and max countour points for nice drawings\n    var minContourPoint = horizontalQ ? {x: 0, y: d3.min(violinPointsValues)} :  {x: d3.min(violinPointsValues), y: 0}\n    var maxContourPoint = horizontalQ ? {x: 0, y: d3.max(violinPointsValues)} :  {x: d3.max(violinPointsValues), y: 0}\n    var violinContourPoints = binned.map(function(bin, i) {\n        return horizontalQ\n        ? {y: (bin.length) ? d3.median(bin): d3.median([bin.x0, bin.x1]), x: frequencies[i]}\n        : {x: (bin.length) ? d3.median(bin): d3.median([bin.x0, bin.x1]), y: frequencies[i]}\n      })\n    // points along which to draw the violin shpe\n    violinContourPoints = [minContourPoint].concat(violinContourPoints).concat([maxContourPoint])\n\n    // set data\n    violinData.binned = binned;\n    violinData.frequencies = frequencies\n    violinData.contour = violinContourPoints\n    violinData.quartiles = pointQuartiles\n    violinData.pointKeys = violinPointsKeys\n    violinData.pointValues = violinPointsValues\n  }\n\n  return calculateViolinValues\n}\n","import {hypenate, safeSelect, round, interpolateColors} from './helpers';\nimport {setupContainer} from './utils';\nimport {colorFunction as CF} from './color-function';\n\n\nexport function numericLegend( selection ) {\n\n  var\n  min=0,\n  max=1,\n  spaceX,\n  spaceY,\n  colorFunction = CF(),\n  namespace='d3sm-linear-vertical-gradient',\n  fontSize = 12,\n  backgroundFill = 'transparent',\n  textColor = 'black',\n  roundTo = 2\n\n\n  legend.min = function(_) { return arguments.length ? (min=_, legend) : min }\n  legend.max = function(_) { return arguments.length ? (max=_, legend) : max }\n  legend.spaceX = function(_) { return arguments.length ? (spaceX=_, legend) : spaceX }\n  legend.spaceY = function(_) { return arguments.length ? (spaceY=_, legend) : spaceY }\n  legend.namespace = function(_) { return arguments.length ? (namespace=_, legend) : namespace }\n  legend.fontSize = function(_) { return arguments.length ? (fontSize=_, legend) : fontSize }\n  legend.backgroundFill = function(_) { return arguments.length ? (backgroundFill=_, legend) : backgroundFill }\n  legend.colorFunction = function(_) { return arguments.length ? (colorFunction=_, legend) : colorFunction }\n  legend.textColor = function(_) { return arguments.length ? (textColor=_, legend) : textColor }\n  legend.roundTo = function(_) { return arguments.length ? (roundTo=_, legend) : roundTo }\n\n  function legend() {\n    // background cliping rectangle\n    var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}\n    var container = setupContainer( selection, namespace, bgcpRect, backgroundFill );\n\n    var defs = safeSelect(selection, 'defs')\n    var linearGradient = safeSelect(defs, 'linearGradient')\n    .attr(\"x1\", \"0%\")\n    .attr(\"y1\", \"100%\")\n    .attr(\"x2\", \"0%\")\n    .attr(\"y2\", \"0%\")\n    .attr('id', hypenate(namespace,'numerical-legend-gradient'))\n\n\n    colorFunction.dataExtent([min, max])\n    .colorBy('value')\n    .valueExtractor(function(k, v, i){return v})\n\n\n    linearGradient.selectAll('stop')\n    .data( colorFunction.colors() )\n    .enter()\n    .append('stop')\n    .attr(\"offset\", function(d, i){ return i / (colorFunction.colors().length - 1) })\n    .attr('stop-color', function(d) {return d})\n\n\n\n\n    var rect = safeSelect(container, 'rect', 'legend')\n    .attr('transform', 'translate(0,'+fontSize+')')\n    .style(\"fill\", \"url(#\"+hypenate(namespace,'numerical-legend-gradient')+\")\")\n    .attr('x', 0)\n    .attr('y', 0)\n    .attr('width', spaceX)\n    .attr('height', spaceY - fontSize*2)\n    .on('mousemove', function(d, i){legendMousemove(d, i, rect, d3.select(this))})\n    .on('mouseout', function(d, i){ d3.select(\"#\"+hypenate(namespace,'legend-tooltip')).remove() })\n\n    var minText = safeSelect(container, 'text', 'min')\n    .text(round(min, 2))\n    .attr('text-anchor', 'middle')\n    .attr(\"font-size\", fontSize+'px')\n    .attr('transform', function(d, i){\n      var\n      x = spaceX / 2,\n      y = spaceY - fontSize / 4,\n      t = 'translate('+x+','+y+')'\n      return t\n    })\n\n    var maxText = safeSelect(container, 'text', 'max')\n    .text(round(max, 2))\n    .attr('text-anchor', 'middle')\n    .attr(\"font-size\", fontSize+'px')\n    .attr('transform', function(d, i){\n      var\n      x = spaceX / 2,\n      y = fontSize,\n      t = 'translate('+x+','+y+')'\n      return t\n    })\n\n\n\n\n  }\n\n  function legendMousemove(d, i, rect, t) {\n    var s = d3.scaleLinear()\n    .domain([0, rect.attr('height')])\n    .range([max, min])\n    var m = d3.mouse(rect.node())\n    var v = round(s(m[1]),roundTo)\n\n    var strokeColor = colorFunction(undefined, v, undefined, 'stroke')\n    var fillColor = colorFunction(undefined, v, undefined, 'fill')\n\n    var div = safeSelect(d3.select('body'), 'div', hypenate(namespace,'legend-tooltip'))\n    .attr('id', hypenate(namespace,'legend-tooltip'))\n    .style('position', 'absolute')\n    .style('left', (d3.event.pageX+15)+'px')\n    .style('top', (d3.event.pageY+15)+'px')\n    .style('background-color', fillColor)\n    .style('border-color', strokeColor)\n\n    .style('min-width', (fontSize * (String(max).split('.')[0].length+3))+'px')\n    .style('min-height', (fontSize * (String(max).split('.')[0].length+3))+'px')\n    .style('border-radius', '50%')\n    .style('border-radius', '5000px')\n\n    .style('display', 'flex')\n    .style('justify-content', 'center')\n    .style('text-align', 'middle')\n    .style('padding', 2+\"px\")\n\n    .style('border-style', 'solid')\n    .style('border-width', 2)\n\n    var text = safeSelect(div, 'div')\n    .text(v)\n    .style('color', textColor)\n    .style('align-self', 'center')\n  }\n\n  return legend\n}\n","import {hypenate, safeSelect, round, interpolateColors} from './helpers';\nimport {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer} from './utils';\nimport {colorFunction as CF} from './color-function';\nimport {groupingSpacer} from './grouping-spacer';\nimport {unique, flatten} from './array-functions';\n\nexport function categoricLegend( selection ) {\n  var\n  categories,\n\n  /**\n  * Data to plot. Assumed to be a object, where each key corresponds to a legend\n  * (see {@link legend#data})\n  * @param {Object} [data=undefined]\n  * @memberof legend#\n  * @property\n  */\n  data,\n  /**\n  * Which direction to render the bars in\n  * (see {@link legend#orient})\n  * @param {number} [orient='horizontal']\n  * @memberof legend#\n  * @property\n  */\n  orient='horizontal',\n  /**\n  * Amount of horizontal space (in pixels) avaible to render the legend in\n  * (see {@link legend#spaceX})\n  * @param {number} [spaceX=undefined]\n  * @memberof legend#\n  * @property\n  */\n  spaceX,\n  /**\n  * Amount of vertical space (in pixels) avaible to render the legend in\n  * (see {@link legend.spaceY})\n  * @param {number} [spaceY=undefined]\n  * @memberof legend#\n  * @property\n  */\n  spaceY,\n\n  /**\n  * Whether or not to allow legend to render elements pass the main spatial dimension\n  * given the orientation (see {@link legend#orient}), where {@link legend#orient}=\"horizontal\"\n  * the main dimension is {@link legend#spaceX} and where {@link legend#orient}=\"vertical\"\n  * the main dimension is {@link legend#spaceY}\n  * @param {boolean} [overflowQ=false]\n  * @memberof legend#\n  * @property\n  */\n  overflowQ = false,\n\n  /**\n  * An array - putatively of other arrays - depicting how bars should be arranged\n  * @param {Array[]} [grouping=undefined]\n  * @memberof legend#\n  * @property\n  */\n  grouping,\n\n  /**\n  * How to get the value of the legend\n  * @param {function} [valueExtractor=function(key, index) { return data[key] }]\n  * @memberof legend#\n  * @property\n  */\n  valueExtractor = function(key, index) { return data[key] },\n  /**\n  * How to sort the bars - if {@link bar#grouping} is not provided.\n  * @param {function} [sortingFunction=function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}]\n  * @memberof bar#\n  * @property\n  */\n  sortingFunction = function(keyA, keyB) {return d3.ascending(keyA, keyB)},\n  /**\n  * Default space for the spacer (percentage) of main dimension given the orientation\n  * (see {@link legend#orient}), where {@link legend#orient}=\"horizontal\"\n  * the main dimension is {@link legend#spaceX} and where {@link legend#orient}=\"vertical\"\n  * the main dimension is {@link legend#spaceY} between bars\n  * @param {number} [objectSpacer=0.05]\n  * @memberof legend#\n  * @property\n  */\n  objectSpacer = 0.05,\n  /**\n  * The minimum size that an object can be\n  * @param {number} [minObjectSize=50]\n  * @memberof legend#\n  * @property\n  */\n  minObjectSize = 10,\n  /**\n  * The maximum size that an object can be\n  * @param {number} [maxObjectSize=100]\n  * @memberof legend#\n  * @property\n  */\n  maxObjectSize = 100,\n\n  /**\n  * The stroke width of the bars\n  * @param {number} [barStrokeWidth=2]\n  * @memberof legend#\n  * @property\n  */\n  bubbleStrokeWidth = 2,\n  /**\n  * Instance of ColorFunction\n  * @param {function} [colorFunction = colorFunction()]\n  * @memberof legend#\n  * @property\n  */\n  colorFunction = CF(),\n\n  /**\n  * Color of the background\n  * @param {string} [backgroundFill=\"transparent\"]\n  * @memberof legend#\n  * @property\n  */\n  backgroundFill = 'transparent',\n  /**\n  * Namespace for all items made by this instance of legend\n  * @param {string} [namespace=\"d3sm-legend\"]\n  * @memberof legend#\n  * @property\n  */\n  namespace = 'd3sm-legend',\n  /**\n  * Class name for legend container (<g> element)\n  * @param {string} [objectClass=\"legend\"]\n  * @memberof legend#\n  * @property\n  */\n  objectClass = 'legend',\n\n  /**\n  * Duration of all transitions of this element\n  * @param {number} [transitionDuration=1000]\n  * @memberof legend#\n  * @property\n  */\n  transitionDuration = 1000,\n  /**\n  * Easing function for transitions\n  * @param {d3.ease} [easeFunc=d3.easeExp]\n  * @memberof legend#\n  * @property\n  */\n  easeFunc = d3.easeExp\n\n\n  legend.categories = function(_) { return arguments.length ? (categories=_, legend) : categories };\n\n  /**\n   * Gets or sets the selection in which items are manipulated\n   * @param {d3.selection} [_=none]\n   * @returns {legend | d3.selection}\n   * @memberof legend\n   * @property\n   * by default selection = selection\n   */\n  legend.selection = function(_) { return arguments.length ? (selection = _, legend) : selection; };\n  /**\n   * Gets or sets the data\n   * (see {@link legend#data})\n   * @param {number} [_=none]\n   * @returns {legend | object}\n   * @memberof legend\n   * @property\n   */\n  legend.data = function(_) { return arguments.length ? (data = _, legend) : data; };\n  /**\n   * Gets or sets the orient of the bars\n   * (see {@link legend#orient})\n   * @param {number} [_=none]\n   * @returns {legend | object}\n   * @memberof legend\n   * @property\n   */\n  legend.orient = function(_) { return arguments.length ? (orient = _, legend) : orient; };\n  /**\n   * Gets or sets the amount of horizontal space in which items are manipulated\n   * (see {@link legend#spaceX})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {legend | number}\n   * @memberof legend\n   * @property\n   * by default spaceX = undefined\n   */\n  legend.spaceX = function(_) { return arguments.length ? (spaceX = _, legend) : spaceX; };\n  /**\n   * Gets or sets the amount of vertical space in which items are manipulated\n   * (see {@link legend#spaceY})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {legend | number}\n   * @memberof legend\n   * @property\n   * by default spaceY = undefined\n   */\n  legend.spaceY = function(_) { return arguments.length ? (spaceY = _, legend) : spaceY; };\n\n  /**\n   * Gets / sets whether or not legend is allowed to go beyond specified dimensions\n   * (see {@link legend#spaceX})\n   * @param {boolean} [_=none]\n   * @returns {legend | boolean}\n   * @memberof legend\n   * @property\n   * by default overflowQ = false\n   */\n  legend.overflowQ = function(_) { return arguments.length ? (overflowQ = _, legend) : overflowQ; };\n  /**\n   * Gets / sets the grouping of the bars\n   * (see {@link legend#grouping})\n   * @param {Array[]} [_=none]\n   * @returns {legend | Array[]}\n   * @memberof legend\n   * @property\n   * by default grouping = undefined\n   */\n  legend.grouping = function(_) { return arguments.length ? (grouping = _, legend) : grouping; };\n  /**\n   * Gets / sets the valueExtractor\n   * (see {@link legend#valueExtractor})\n   * @param {function} [_=none]\n   * @returns {legend | function}\n   * @memberof legend\n   * @property\n   * by default valueExtractor = function(key, index) { return data[key] },\n   */\n  legend.valueExtractor = function(_) { return arguments.length ? (valueExtractor = _, legend) : valueExtractor; };\n  /**\n   * Gets / sets the sortingFunction\n   * (see {@link bar#sortingFunction})\n   * @param {function} [_=none]\n   * @returns {bar | function}\n   * @memberof bar\n   * @property\n   * by default sortingFunction = function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])},\n   */\n  legend.sortingFunction = function(_) { return arguments.length ? (sortingFunction = _, legend) : sortingFunction; };\n  /**\n   * Gets / sets objectSpacer\n   * (see {@link legend#objectSpacer})\n   * @param {number} [_=none]\n   * @returns {legend | number}\n   * @memberof legend\n   * @property\n   * by default objectSpacer = 0.05\n   */\n  legend.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, legend) : objectSpacer; };\n  /**\n   * Gets / sets the minObjectSize\n   * (see {@link legend#minObjectSize})\n   * @param {number} [_=none]\n   * @returns {legend | number}\n   * @memberof legend\n   * @property\n   * by default minObjectSize = 50\n   */\n  legend.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, legend) : minObjectSize; };\n  /**\n   * Gets / sets the maxObjectSize\n   * (see {@link legend#maxObjectSize})\n   * @param {number} [_=none]\n   * @returns {legend | number}\n   * @memberof legend\n   * @property\n   * by default maxObjectSize = 100\n   */\n  legend.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, legend) : maxObjectSize; };\n\n  /**\n   * Gets / sets the barStrokeWidth\n   * (see {@link legend#barStrokeWidth})\n   * @param {number} [_=none]\n   * @returns {legend | number}\n   * @memberof legend\n   * @property\n   * by default barStrokeWidth = 2\n   */\n  legend.bubbleStrokeWidth = function(_) { return arguments.length ? (bubbleStrokeWidth = _, legend) : bubbleStrokeWidth; };\n  /**\n   * Gets / sets the colorFunction\n   * (see {@link legend#colorFunction})\n   * @param {number} [_=none]\n   * @returns {legend | number}\n   * @memberof legend\n   * @property\n   * by default colorFunction = colorFunction()\n   */\n  legend.colorFunction = function(_) { return arguments.length ? (colorFunction = _, legend) : colorFunction; };\n\n  /**\n   * Gets / sets the backgroundFill\n   * (see {@link legend#backgroundFill})\n   * @param {string} [_=none]\n   * @returns {legend | string}\n   * @memberof legend\n   * @property\n   * by default backgroundFill = 'transparent'\n   */\n  legend.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, legend) : backgroundFill; };\n  /**\n   * Gets / sets the namespace\n   * (see {@link legend#namespace})\n   * @param {string} [_=none]\n   * @returns {legend | string}\n   * @memberof legend\n   * @property\n   * by default namespace = 'd3sm-legend'\n   */\n  legend.namespace = function(_) { return arguments.length ? (namespace = _, legend) : namespace; };\n  /**\n   * Gets / sets the objectClass\n   * (see {@link legend#objectClass})\n   * @param {string} [_=none]\n   * @returns {legend | string}\n   * @memberof legend\n   * @property\n   * by default objectClass = 'tick-group'\n   */\n  legend.objectClass = function(_) { return arguments.length ? (objectClass = _, legend) : objectClass; };\n  /**\n   * Gets / sets the transitionDuration\n   * (see {@link legend#transitionDuration})\n   * @param {number} [_=none]\n   * @returns {legend | number}\n   * @memberof legend\n   * @property\n   * by default transitionDuration = 1000\n   */\n  legend.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, legend) : transitionDuration; };\n  /**\n   * Gets / sets the easeFunc\n   * (see {@link legend#easeFunc})\n   * @param {d3.ease} [_=none]\n   * @returns {legend | d3.ease}\n   * @memberof legend\n   * @property\n   * by default easeFunc = d3.easeExp\n   */\n  legend.easeFunc = function(_) { return arguments.length ? (easeFunc = _, legend) : easeFunc; };\n\n\n  function legend() {\n    var horizontalQ = (orient == 'horizontal') ? true : false\n    var verticalQ = !horizontalQ\n    // background cliping rectangle\n    var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}\n    var container = setupContainer( selection, namespace, bgcpRect, backgroundFill );\n\n\n    colorFunction.dataExtent([0, categories.length - 1])\n    .colorBy('categories')\n    .categoryExtractor(function(k, v, i){return v})\n\n    var r = Math.min(spaceX, spaceY) / 2\n    var numberOfObjects = categories.length\n\n    // if grouping is undefined sort barKeys by sortingFunction\n    var ordered = (grouping == undefined) ? categories.sort(sortingFunction) : grouping\n    // ordered might be nested depending on grouping\n    var catKeys = flatten(ordered)\n\n    var space = horizontalQ ? spaceX : spaceY\n    // calculate object size\n    var objectSize = calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ)\n    // calculate spacer size if needed\n    var spacerSize = calculateWidthOfSpacer(catKeys, space, objectSize, numberOfObjects, objectSpacer, overflowQ)\n    // make the nested groups\n    var spacerFunction = groupingSpacer()\n    .horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects)\n    .objectClass(objectClass).objectSize(objectSize).spacerSize(spacerSize)\n    .transitionDuration(transitionDuration).easeFunc(easeFunc)\n    .namespace(namespace)\n\n    spacerFunction(container, ordered, 0)\n    var r = Math.min(objectSize, spaceX, spaceY) / 2 - bubbleStrokeWidth\n\n    container.selectAll('g:not(.to-remove).'+objectClass).each(function(cat, i) {\n      var t = d3.select(this)\n      var c = safeSelect(t, 'circle')\n      var fillColor = colorFunction(undefined, cat, i, 'fill'), // prevent duplicate computation\n      strokeColor = colorFunction(undefined, cat, i,  'stroke')\n\n      var cx = horizontalQ\n        ? r+bubbleStrokeWidth\n        : (spaceX - r*2) / 2 + r\n      var cy = verticalQ\n        ? r+bubbleStrokeWidth\n        : (spaceX - r*2) / 2 + r\n\n      c.attr(\"r\", r)\n      .attr('cx', cx)\n      .attr('cy', cy)\n      .attr('fill', fillColor)\n      .attr('stroke', strokeColor)\n      .attr('stroke-width', bubbleStrokeWidth)\n\n      var text = safeSelect(t, 'text')\n      text.text(cat)\n      .attr('text-anchor', 'middle')\n      .attr(\"transform\", function(d, i){\n        var\n        x = cx,\n        y = cy + text.node().getBoundingClientRect().height / 4,\n        t = 'translate('+x+','+y+')'\n        return t\n      })\n\n    })\n\n\n  }\n\n  return legend\n}\n","import {hypenate, safeSelect, modifyHexidecimalColorLuminance, extractViolinValues, quartiles} from './helpers';\nimport {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer} from './utils';\nimport {unique, hasQ, flatten, whichBin} from './array-functions';\nimport {groupingSpacer} from './grouping-spacer';\nimport {colorFunction as CF} from './color-function';\nimport {tooltip as TTip} from './tooltip';\nimport {lasso} from './lasso';\nimport './d3-prototypes';\n\nexport function lassoWidget( selection ) {\n  var\n  namespace = 'd3sm-lasso',\n  selection = selection,\n  svg,\n  chartContainer,\n  objectContainer,\n  objectClass,\n  lassoLine = d3.line()\n  .x(function(d, i){return d[0]})\n  .y(function(d, i){return d[1]})\n  .curve(d3.curveLinearClosed),\n  path,\n  colorFunction = CF(),\n  maxNumberOfGroups,\n  dataExtractor,\n  xScale,\n  yScale\n\n  function onError(msg, style){\n    console.log(msg, style)\n  }\n  // setup the container\n  setupDataselectContainer()\n\n  lassoWidget.objectClass = function(_){return arguments.length ? (objectClass='.'+_.replace('.',''), lassoWidget) : objectClass}\n  lassoWidget.svg = function(_){return arguments.length ? (svg=_, lassoWidget) : svg}\n  lassoWidget.submit = function(_){return arguments.length ? (submit=_, lassoWidget) : submit}\n  lassoWidget.maxNumberOfGroups = function(_){return arguments.length ? (maxNumberOfGroups=_, lassoWidget) : maxNumberOfGroups}\n  lassoWidget.onError = function(_){return arguments.length ? (onError=_, lassoWidget) : onError}\n  lassoWidget.objectContainer = function(_){return arguments.length ? (objectContainer=_, lassoWidget) : objectContainer}\n  lassoWidget.chartContainer = function(_){return arguments.length ? (chartContainer=_, lassoWidget) : chartContainer}\n  lassoWidget.dataExtractor = function(_){return arguments.length ? (dataExtractor=_, lassoWidget) : dataExtractor}\n  lassoWidget.xScale = function(_){return arguments.length ? (xScale=_, lassoWidget) : xScale}\n  lassoWidget.yScale = function(_){return arguments.length ? (yScale=_, lassoWidget) : yScale}\n\n  function styles() {\n    var s = d3.select(\"html\").select(\"style.\"+namespace+'lasso-widget')\n    if (s.empty()) {\n      d3.select(\"html\").append(\"style\")\n      .classed(namespace+'lasso-widget', true)\n      .html(\n        \".\"+hypenate(namespace, \"data-table\") + \"{\\\n          height:100px;\\\n          overflow:auto;\\\n        }\"\n      )\n    }\n  }\n\n\n  function lassoWidget() {\n    styles()\n\n  }\n\n  function submit(data) {\n    console.log(data)\n  }\n\n  function plusTab(tabList) {\n    var tab = safeSelect(tabList, 'li', hypenate(namespace, 'plus-tab'))\n    .classed('ml-auto', true)\n    .classed('nav-item', true)\n    .on('click', makeNewGroup),\n\n    anchor = safeSelect(tab, 'a', 'nav-link'),\n    icon = safeSelect(anchor, 'i', 'fa')\n    .classed('fa-plus fa-2x text-success', true)\n  }\n\n  function sendTab(tabList) {\n    var tab = safeSelect(tabList, 'li', hypenate(namespace, 'send-tab'))\n    .classed('ml-auto', true)\n    .classed('nav-item', true)\n    .on('click', clickSend)\n    ,\n\n    anchor = safeSelect(tab, 'a', 'nav-link'),\n    icon = safeSelect(anchor, 'i', 'fa')\n    .classed('fa-paper-plane-o fa-2x text-primary', true)\n  }\n\n  function clickSend() {\n    var data = gatherDataLists()\n    submit(data)\n  }\n\n  function closeTab(tabList) {\n    var tab = safeSelect(tabList, 'li', hypenate(namespace, 'close-tab'))\n    .classed('ml-auto', true)\n    .classed('nav-item', true)\n    .on('click', function(){\n      var m = d3.mouse(d3.select(\"html\").node())\n      selection.select('div.card').style(\"position\", 'relative')\n      .transition().duration(1000)\n      .ease(d3.easeBack)\n      .style('left', window.outerWidth +'px')\n      .remove()\n    })\n    ,\n\n    anchor = safeSelect(tab, 'a', 'nav-link'),\n    icon = safeSelect(anchor, 'i', 'fa')\n    .classed('fa-window-close-o fa-2x text-danger', true)\n  }\n\n\n\n  function setupDataselectContainer() {\n    var\n    card = safeSelect(selection, 'div', 'card'),\n    cardHeader = safeSelect(card, 'div', 'card-header'),\n\n    tabList = safeSelect(cardHeader, 'ul', 'nav')\n    .classed('nav-tabs card-header-tabs', true)\n    .classed(hypenate(namespace, 'tab-list'), true)\n    .attr('role', 'tablist'),\n\n    cardBody = safeSelect(card, 'div', 'card-body'),\n    tabContent = safeSelect(cardBody, 'div', 'tab-content')\n    .classed(hypenate(namespace, 'tab-content'), true),\n\n    // leftTabs = safeSelect(tabList, 'div', 'nav mr-auto left-aligned-tabs')\n    rightTabs = safeSelect(tabList, 'div', 'right-aligned-tabs').classed('nav ml-auto ', true)\n\n    plusTab(rightTabs)\n    sendTab(rightTabs)\n    closeTab(rightTabs)\n    var\n    defaultTab = safeSelect(tabContent, 'div', 'tab-pane')\n    .classed(hypenate(namespace,'default-tab'), true)\n    .classed(\"active\", true)\n    .classed('text-left', true),\n\n    p = safeSelect(defaultTab, 'div')\n    .html(\n      \"Click <kbd><i class='fa fa-plus text-success'></i></kbd> to add a new group.<br>\"+\n      \"Click <kbd><i class='fa fa-paper-plane-o text-primary'></i></kbd> to submit for re-analysis.<br>\"+\n      \"Click <kbd><i class='fa fa-window-close-o text-danger'></i></kbd> to close the dataselect widget.\"\n    )\n    // .text('Click the green plus to add a new group.')\n  }\n\n\n  function makeRemoveButton(paneBtnList) {\n    var\n    btn = safeSelect(paneBtnList, 'button', 'remove-btn')\n    .classed('btn btn-danger', true)\n    .on('click', removeBtnClickToRemove),\n    i = safeSelect(btn, 'i', 'fa').classed('fa-trash-o', true),\n    s = safeSelect(btn, 'span').text('Remove group')\n    return btn\n  }\n\n  function removeBtnClickToRemove(d, i) {\n    var\n    tab = selection.select('#'+hypenate(namespace,'tab',d)),\n    pane = selection.select('#'+hypenate(namespace,'tab','pane',d))\n\n    tab.remove()\n    pane.remove()\n\n    showRemainingPanes()\n    updateTabAndPaneNumbers()\n  }\n\n\n  function togglePaneById(id) {\n    var panes = selection.select('.'+hypenate(namespace, 'tab-content'))\n    panes.selectAll('div.tab-pane')\n    .each(function(d, i){\n      d3.select(this).classed('active show', d3.select(this).attr('id') == id)\n    })\n  }\n\n  function insertTab(tabs, n) {\n\n    // var li = tabs.insert(\"li\", ':nth-child('+(n)+')')\n    var li = tabs.insert(\"li\", '.right-aligned-tabs')\n    .classed('nav-item', true)\n    .classed('alert', true)\n    .classed('alert-secondary', true)\n    .classed('alert-dismissable', true)\n    .classed('fade', true)\n    .classed('show', true)\n    // .classed('mr-auto', true)\n    .attr(\"role\", 'alert')\n    .attr(\"id\", hypenate(namespace,'tab',n))\n    .classed(hypenate(namespace,'tab'), true)\n\n    var a = safeSelect(li, 'a')\n    .attr('data-toggle', 'tab')\n    .text('Group ' + n)\n    .attr('href', '#'+hypenate(namespace,'tab','pane',n))\n    .on('dblclick', function(d, i){\n      var t = d3.select(this)\n      t.attr('contenteditable', true)\n      d3.select(t.node().parentNode)\n      .classed('alert-secondary', false)\n      .classed('alert-warning', true)\n\n      // d3.select(\"html\").on(\"click\", dispatchBlur)\n      //\n      // function dispatchBlur(d, i) {\n      //   if (d3.event.target != d3.select(this)) {\n      //     d3.select(this).dispatch(\"blur\")\n      //   }\n      // }\n\n\n    })\n    .on('blur', function(d, i){\n\n      var t = d3.select(this)\n      t.attr('contenteditable', false)\n      d3.select(t.node().parentNode)\n      .classed('alert-secondary', true)\n      .classed('alert-warning', false)\n\n    })\n    .on('input', function(d, i){\n      var t = d3.select(this)\n      var txt = t.text()\n      d3.select(t.attr('href')).select(\"p.lead\").text(txt)\n    })\n\n\n    return li\n  }\n\n  function makeTabPane(panes, n) {\n    var pane = panes.append('div')\n    .datum(n)\n    .attr('role', 'tabpanel')\n    .classed('tab-pane', true)\n    .classed('text-left', true)\n    .classed(hypenate(namespace,'tab','pane'), true)\n    .attr('id', hypenate(namespace,'tab','pane',n))\n    return pane\n  }\n\n  function populatePane(pane, n) {\n    var\n    lead = safeSelect(pane, 'p', 'lead').text('Group '+n),\n\n    tableContainer = safeSelect(pane, 'div', 'table-responsive')\n    .attr(\"class\", hypenate(namespace, \"data-table\")),\n\n    table = safeSelect(tableContainer, \"table\", \"table\")\n    .classed(\"table-sm\", true).classed(\"table-hover\", true),\n\n    caption = safeSelect(table, \"caption\", \"caption\").html(\"List of selected\"),\n\n    btns = safeSelect(pane, 'div', 'text-right'),\n\n    cN = n % colorFunction.colors().length\n    if (n % 2 != 0) { cN = (colorFunction.colors().length - 1) - cN }\n\n\n\n\n\n    var lassoBtn = makeLassoButton(btns)\n    var currentLasso = makeLassoFunction(n, colorFunction.colors()[cN])\n    var clearBtn = makeClearButton(btns)\n\n    bindButtons(lassoBtn, clearBtn, currentLasso, table)\n\n    var removeBtn = makeRemoveButton(btns)\n\n    lead.on(\"mouseover\", function(){\n      currentLasso.draw()\n    })\n    lead.on(\"mouseout\", function(){\n      currentLasso.remove()\n    })\n\n  }\n\n  function makeLassoButton(btns) {\n    var btn = safeSelect(btns, 'button', 'lasso-btn')\n    .classed('btn btn-info', true)\n    .classed(namespace, true)\n    // .datum([lasso])\n    var i = safeSelect(btn, 'i', 'fa').classed('fa-hand-pointer-o', true)\n    var s = safeSelect(btn, 'span').text('Lasso select')\n    return btn\n\n  }\n\n  function makeClearButton(btns) {\n    var btn = safeSelect(btns, 'button', 'clear-btn')\n    .classed('btn btn-light', true)\n    .classed(namespace, true)\n    var i = safeSelect(btn, 'i', 'fa').classed('fa-eraser', true)\n    var s = safeSelect(btn, 'span').text('Clear selection')\n    return btn\n  }\n\n  function makeLassoFunction(instance, color) {\n    var currentLasso = lasso()\n    .namespace(namespace)\n    .svg(svg)\n    .objectClass(objectClass)\n    .chartContainer(chartContainer)\n    .objectContainer(objectContainer)\n    .instance(instance)\n    .color(color)\n    .yScale(yScale)\n    .xScale(xScale)\n\n    return currentLasso\n  }\n\n  function bindButtons(lassoBtn, clearBtn, currentLasso, table) {\n    currentLasso.eventCatcher(lassoBtn)\n    lassoBtn.node().addEventListener(\"click\", lassoBtnToggle)\n\n    lassoBtn.datum([currentLasso])\n    .on(hypenate(namespace,'render'), function(){\n        d3.select(this).datum()[0].draw()\n    })\n    .on(hypenate(namespace,\"drag\"), function(las){\n      var nodes = chartContainer.selectAll(\".in-lasso-\"+las[0].instance()).nodes()\n      var tableData = nodes.map(function(d, i){\n        var extracted = dataExtractor(d3.select(d).datum())\n        extracted[\"__node\"] = d\n        return extracted\n      })\n\n\n      makeTable(table, tableData, currentLasso)\n    })\n\n    clearBtn.on('click', function(){\n      currentLasso.allPoints([])\n      currentLasso.currentPoints([])\n      lassoBtn.dispatch(hypenate(namespace,\"drag\"))\n\n    })\n\n  }\n\n  function lassoBtnToggle() {\n    var that = d3.select(this)\n    var las = that.datum()[0]\n\n    las.toggle()\n    var activeQ = las.activeQ()\n\n    if (las.activeQ()) {\n      that.classed('btn-info', !activeQ)\n      that.classed('btn-warning', activeQ)\n      that.select(\"span\").text(\"Lasso select (active)\")\n      selection.selectAll(\".\"+namespace+\".lasso-btn\").dispatch(hypenate(namespace,'render'))\n      d3.select(\"html\").node().addEventListener('mousedown', monitorLassoButtonState)\n    } else {\n      that.classed('btn-info', !activeQ)\n      that.classed('btn-warning', activeQ)\n      that.select(\"span\").text(\"Lasso select\")\n      d3.select(\"html\").node().removeEventListener('mousedown', monitorLassoButtonState)\n    }\n\n    function monitorLassoButtonState(event) {\n      /*\n      activeLasso stops event stopPropagation, so this event will not register in\n      that case. Thus we only need to ensure that the usre is clicking on the lasso button\n      otherwise, de-spawn lasso.\n      */\n      if (\n        event.target != that.node() &&\n        event.target != that.select(\"span\").node() &&\n        event.target != that.select(\"i\").node()\n      ) {\n        // event.preventDefault()\n        // event.stopPropagation()\n        las.toggle(false)\n        that.classed('btn-info', true)\n        that.classed('btn-warning', false)\n        that.select(\"span\").text(\"Lasso select\")\n        // console.log(that, that.node())\n        // that.dispatch(\"focusout\")\n      }\n    }\n\n  }\n\n\n  function updateTableHeaderColumns(headR, tableData) {\n    var headerKeys = d3.keys(tableData[0]).filter(k=>k!=\"__node\")\n    if (headerKeys.length > 0) {\n      // headerKeys = [\"remove\"].concat(headerKeys)\n      headerKeys.push(\"remove\")\n    }\n\n    var headerCols = headR.selectAll(\"th\")\n\n    headerCols = headerCols.data(headerKeys)\n    headerCols.exit().remove()\n    headerCols = headerCols.merge(headerCols.enter().append(\"th\").attr(\"scope\",\"col\"))\n    .text(function(d, i){return d})\n\n    return headerKeys\n  }\n\n  function updateTableRows(body, tableData) {\n    var bodyRows = body.selectAll(\"tr\")\n\n    bodyRows = bodyRows.data(tableData)\n    bodyRows.exit().remove()\n    bodyRows = bodyRows.merge(bodyRows.enter().append(\"tr\"))\n\n    return bodyRows\n  }\n\n  function updateTableRowColumns(cols, headerKeys, rowData, tableData, lasso, table) {\n    cols = cols.data(headerKeys)\n    cols.exit().remove()\n    cols = cols.merge(cols.enter().append(\"td\"))\n\n    cols.html(function(d, i){\n      if (d != \"remove\") { return rowData[d] }\n      return \"<i class='fa fa-close'></i>\"\n    }).classed(\"text-left\", function(d, i){\n      if (d != \"remove\") { return false }\n      return true\n    })\n    // .attr(\"scope\",function(d, i){\n    //   if (d != \"remove\") { return false }\n    //   return true\n    // })\n\n    cols.select(\"i.fa-close\").on(\"click\", function(d, i){\n      var node = rowData[\"__node\"],\n      n = d3.select(node)\n      n.classed(\"in-lasso\", false)\n      n.classed(\"in-lasso-\"+lasso.instance(), false)\n\n      tableData.map(function(dd, j){\n        if (dd[\"__node\"] == node) {\n          tableData.splice(j, 1)\n          makeTable(table, tableData, lasso)\n        }\n      })\n\n    })\n  }\n\n  function makeTable(table, tableData, lasso) {\n    var\n    head = safeSelect(table, \"thead\"),\n    headR = safeSelect(head, \"tr\"),\n    body = safeSelect(table, \"tbody\"),\n\n    headerKeys = updateTableHeaderColumns(headR, tableData),\n\n    bodyRows = updateTableRows(body, tableData)\n\n    bodyRows.each(function(rowData, i){\n      var t = d3.select(this)\n      var cols = t.selectAll(\"td\")\n\n      updateTableRowColumns(cols, headerKeys, rowData, tableData, lasso, table)\n\n      t.on(\"mouseover\", function(d, i) {\n        lasso.applyObjectAttributes(d3.select(d[\"__node\"]),true)\n      })\n      .on(\"mouseout\", function(d, i) {\n        lasso.applyObjectAttributes(d3.select(d[\"__node\"]),false)\n      })\n    })\n\n\n  }\n\n\n\n\n\n\n\n  function makeNewGroup(d, i) {\n\n    var tabs = selection.select('.'+hypenate(namespace, 'tab-list')),\n    panes = selection.select('.'+hypenate(namespace, 'tab-content')),\n    n = newGroupNumber()\n\n    if (tabs.selectAll('.'+hypenate(namespace,'tab')).size() == maxNumberOfGroups) {\n      onError('only '+maxNumberOfGroups+' allowed.', 'warning')\n      return\n    }\n\n    var\n    tab = insertTab(tabs, n),\n    pane = makeTabPane(panes, n)\n\n    populatePane(pane, n)\n    togglePaneById(pane.attr('id'))\n\n  }\n\n  function newGroupNumber() {\n    var tabs = selection.select('.'+hypenate(namespace, 'tab-list'))//,\n    var\n    n = tabs.selectAll('li').size() - 3, // minus 1 for plus tab\n    s = 'Group ' + n,\n    gs = []\n    tabs.each(function(d, i){ return gs.push(d3.select(this).select(\"a\").text()) })\n    while (gs.includes(s)) { n+=1; s = 'Group ' + n; }\n    return n\n  }\n\n  function updateTabAndPaneNumbers() {\n    var\n    tabs = selection.select('.'+hypenate(namespace, 'tab-list')),\n    panes = selection.select('.'+hypenate(namespace, 'tab-content'))\n\n    tabs = tabs.selectAll('.'+hypenate(namespace,'tab'))\n    panes = panes.selectAll('.'+hypenate(namespace,'tab', 'pane'))\n\n    tabs.each(function(d, i){\n      d3.select(this).datum(i)\n      .attr(\"id\", hypenate(namespace,'tab',i))\n      .select('a')\n      .attr('href', '#'+hypenate(namespace,'tab','pane',i))\n      .text(function(dd, ii){\n        var curText = d3.select(this).text();\n        if (curText.split(' ')[0] == 'Group') {\n            return 'Group ' + i\n        }\n        return curText\n      })\n    })\n\n    panes.each(function(d, i){\n      d3.select(this).datum(i)\n      .attr('id', hypenate(namespace,'tab','pane',i))\n      safeSelect(d3.select(this), 'p', 'lead')\n      .text(function(dd, ii){\n        var curText = d3.select(this).text();\n        if (curText.split(' ')[0] == 'Group') {\n            return 'Group ' + i\n        }\n        return curText\n      })\n      safeSelect(d3.select(this), 'button', 'remove-btn')\n      .on('click', removeBtnClickToRemove)\n    })\n\n  }\n\n  function gatherDataLists() {\n    var tabs = selection.select('.'+hypenate(namespace, 'tab-list'))\n    var panes = selection.select('.'+hypenate(namespace, 'tab-content'))\n    var tables = panes.selectAll('.'+hypenate(namespace, \"data-table\"))\n    var data = {}\n\n    var textGroups = tabs.selectAll('li.'+hypenate(namespace,'tab') + ' > a')\n    .nodes().map(function(d, i){return d3.select(d).text()})\n\n    textGroups.map(function(e, i){\n      data[e] = []\n    })\n\n\n    tables.each(function(d, i){\n      var table = d3.select(this).select(\"tbody\")\n      data[textGroups[i]] = table.selectAll('tr').data()\n    })\n\n    return data\n  }\n\n  function showRemainingPanes() {\n    var\n    tabs = selection.select('.'+hypenate(namespace, 'tab-list')),\n    panes = selection.select('.'+hypenate(namespace, 'tab-content')),\n    remainingTabs = tabs.selectAll('.'+hypenate(namespace,'tab'))\n\n    if (remainingTabs.size() == 0) {\n      panes.select('.'+hypenate(namespace,'default-tab'))\n      .classed(\"active\", true)\n      .classed('text-left', true)\n    }\n    else {\n      var lastTab = remainingTabs.nodes()[remainingTabs.size()-1],\n      lastPaneId = d3.select(lastTab).select('a').attr('href')\n      panes.select(lastPaneId)\n      .classed(\"active\", true)\n      .classed('text-left', true)\n    }\n  }\n\n\n  return lassoWidget\n}\n","import {hypenate, safeSelect} from './helpers';\nimport {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer, log} from './utils';\nimport {unique, flatten} from './array-functions';\nimport {groupingSpacer} from './grouping-spacer';\nimport {colorFunction as CF} from './color-function';\nimport {tooltip as TTip} from './tooltip';\nexport function upset ( selection ) {\n  var\n  data,\n  orient='horizontal',\n  spaceX,\n  spaceY,\n  overflowQ=false,\n  minObjectSize=20,\n  maxObjectSize=50,\n  circleStrokeWidth=2,\n  // colorFunction=\n  backgroundFill = 'transparent',\n  namespace='d3sm-upset',\n  objectClass = 'upset',\n\n  transitionDuration = 1000,\n  easeFunc = d3.easeExp,\n\n  setKey = \"set\",\n  intersectionKey = \"intersection\",\n  elementsKey = \"elements\",\n\n  setExtractor = function(key, i) {return data[key][setKey]},\n  intersectionExtractor = function(key, i) {return data[key][intersectionKey]},\n  elementExtractor = function(key, i) {return data[key][elementsKey]},\n\n  cellKeys,\n  setValues,\n  intersectionValues,\n  elementValues,\n\n  xObjectSpacer = 0.05,\n  yObjectSpacer = 0.05,\n  radius,\n\n  // listDelim = ';'\n\n  yObjectSize,\n  ySpacerSize,\n  xObjectSize,\n  xSpacerSize,\n\n  setKeySortingFunction = function(a, b) { return setValues.indexOf(setExtractor(a)) - setValues.indexOf(setExtractor(b)) },\n  intersectionKeySortingFunction = function(a, b) { return intersectionValues.indexOf(intersectionExtractor(a)) - intersectionValues.indexOf(intersectionExtractor(b)) }\n\n\n  upset.selection = function(_) { return arguments.length ? (selection = _, upset) : selection; };\n  upset.data = function(_) { return arguments.length ? (data = _, upset) : data; };\n  upset.orient = function(_) { return arguments.length ? (orient = _, upset) : orient; };\n  upset.spaceX = function(_) { return arguments.length ? (spaceX = _, upset) : spaceX; };\n  upset.spaceY = function(_) { return arguments.length ? (spaceY = _, upset) : spaceY; };\n  upset.overflowQ = function(_) { return arguments.length ? (overflowQ = _, upset) : overflowQ; };\n  upset.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, upset) : minObjectSize; };\n  upset.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, upset) : maxObjectSize; };\n  upset.circleStrokeWidth = function(_) { return arguments.length ? (circleStrokeWidth = _, upset) : circleStrokeWidth; };\n  upset.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, upset) : backgroundFill; };\n  upset.namespace = function(_) { return arguments.length ? (namespace = _, upset) : namespace; };\n  upset.objectClass = function(_) { return arguments.length ? (objectClass = _, upset) : objectClass; };\n  upset.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, upset) : transitionDuration; };\n  upset.easeFunc = function(_) { return arguments.length ? (easeFunc = _, upset) : easeFunc; };\n  upset.cellKeys = function(_) { return arguments.length ? (cellKeys = _, upset) : cellKeys; };\n  upset.setValues = function(_) { return arguments.length ? (setValues = _, upset) : setValues; };\n  upset.intersectionValues = function(_) { return arguments.length ? (intersectionValues = _, upset) : intersectionValues; };\n  upset.xObjectSpacer = function(_) { return arguments.length ? (xObjectSpacer = _, upset) : xObjectSpacer; };\n  upset.yObjectSpacer = function(_) { return arguments.length ? (yObjectSpacer = _, upset) : yObjectSpacer; };\n  upset.radius = function(_) { return arguments.length ? (radius = _, upset) : radius; };\n  upset.setExtractor = function(_) { return arguments.length ? (setExtractor = _, upset) : setExtractor; };\n  upset.intersectionExtractor = function(_) { return arguments.length ? (intersectionExtractor = _, upset) : intersectionExtractor; };\n  upset.elementExtractor = function(_) { return arguments.length ? (elementExtractor = _, upset) : elementExtractor; };\n  upset.setKeySortingFunction = function(_) { return arguments.length ? (setKeySortingFunction = _, upset) : setKeySortingFunction; };\n  upset.intersectionKeySortingFunction = function(_) { return arguments.length ? (intersectionKeySortingFunction = _, upset) : intersectionKeySortingFunction; };\n\n  upset.yObjectSize = function(_) { return arguments.length ? (yObjectSize = _, upset) : yObjectSize; };\n  upset.ySpacerSize = function(_) { return arguments.length ? (ySpacerSize = _, upset) : ySpacerSize; };\n  upset.xObjectSize = function(_) { return arguments.length ? (xObjectSize = _, upset) : xObjectSize; };\n  upset.xSpacerSize = function(_) { return arguments.length ? (xSpacerSize = _, upset) : xSpacerSize; };\n\n  function upset() {\n    // for convenience in handling orientation specific values\n    var horizontalQ = (orient == 'horizontal') ? true : false\n    var verticalQ = !horizontalQ\n\n    // background cliping rectangle\n    var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}\n    var container = setupContainer( selection, namespace, bgcpRect, backgroundFill );\n\n\n    cellKeys = d3.keys(data)\n    setValues = unique(cellKeys.map(setExtractor)).sort()\n    intersectionValues = unique(cellKeys.map(intersectionExtractor)).sort().sort(function(a, b){\n      return a.split(';').length - b.split(';').length\n    })\n\n    if (!horizontalQ) {\n      cellKeys.sort(function(a, b){ return setKeySortingFunction(a, b) || intersectionKeySortingFunction(a, b) })\n    } else {\n      cellKeys.sort(function(a, b){ return intersectionKeySortingFunction(a, b) || setKeySortingFunction(a, b) })\n    }\n\n\n\n\n    var\n    xValues = horizontalQ ? intersectionValues : setValues,\n    yValues = horizontalQ ? setValues : intersectionValues,\n    xDim = horizontalQ ? xValues.length : yValues.length,\n    yDim = horizontalQ ? yValues.length : xValues.length\n\n    // console.log(xValues, yValues)\n\n\n    xObjectSize = calculateWidthOfObject(spaceX, xDim, minObjectSize, maxObjectSize, xObjectSpacer, overflowQ)\n    yObjectSize = calculateWidthOfObject(spaceY, yDim, minObjectSize, maxObjectSize, yObjectSpacer, overflowQ)\n    xSpacerSize = calculateWidthOfSpacer(xValues, spaceX, xObjectSize, xDim, xObjectSpacer, overflowQ)\n    ySpacerSize = calculateWidthOfSpacer(yValues, spaceY, yObjectSize, yDim, yObjectSpacer, overflowQ)\n\n    var ySpacer = groupingSpacer()\n    .horizontalQ(false)\n    .moveby('category').numberOfObjects(yDim)\n    .objectSize(yObjectSize).spacerSize(ySpacerSize)\n    .transitionDuration(transitionDuration).easeFunc(easeFunc)\n\n    var xSpacer = groupingSpacer()\n    .horizontalQ(true)\n    .moveby('category').numberOfObjects(xDim)\n    .objectClass(objectClass)\n    .objectSize(xObjectSize).spacerSize(xSpacerSize)\n    .transitionDuration(transitionDuration).easeFunc(easeFunc)\n\n\n\n    if (verticalQ) {\n      xSpacer.objectClass(objectClass)\n      ySpacer.namespace('across').objectClass(hypenate(objectClass, 'across'))\n\n      ySpacer(container, yValues, 0)\n      container.selectAll('g.'+hypenate(objectClass, 'across'))\n      .each(function(d, i){ xSpacer(d3.select(this), xValues, 0) })\n    } else {\n      xSpacer.namespace('across').objectClass(hypenate(objectClass, 'across'))\n      ySpacer.objectClass(objectClass)\n\n      xSpacer(container, xValues, 0)\n      container.selectAll('g.'+hypenate(objectClass, 'across'))\n      .each(function(d, i){ ySpacer(d3.select(this), yValues, 0) })\n    }\n\n\n    var cells = container.selectAll('g:not(.to-remove).'+objectClass)\n    var lookup = {}\n    cellKeys.map(function(k, i){\n      lookup[setExtractor(k)+'::'+intersectionExtractor(k)] = k\n    })\n\n    // var positionedCellKeys = []\n    // for (var i = 0; i < setValues.length; i++) {\n    //   for (var j = 0; j < intersectionValues.length; j++) {\n    //     var lookupValue = lookup[setValues[j]+\"::\"+intersectionValues[i]]\n    //     if (lookupValue == undefined) {\n    //       positionedCellKeys.push(undefined)\n    //     } else {\n    //       positionedCellKeys.push(lookupValue)\n    //     }\n    //     console.log(i, j, lookupValue)\n    //   }\n    // }\n    //\n    // // console.log(positionedCellKeys)\n    //\n    // cells.data(positionedCellKeys);\n\n\n    cells.data(cellKeys);\n\n    cells.each(function(key, i) {\n      var t = d3.select(this)\n      if (key == undefined) {return }\n      var\n      currentData = data[key]\n      // console.log(key, currentData)\n      var\n      set = setExtractor(key, i),\n      intersection = intersectionExtractor(key, i)\n\n      // console.log(set, intersection)\n\n      t.classed(intersection, true)\n      t.classed(set, true)\n\n      var c = safeSelect(t, 'circle', hypenate(objectClass,'circle'))\n      c.attr('cx', xObjectSize / 2)\n      .attr('cy', yObjectSize / 2 )\n      .attr('r', radius == undefined ? Math.min(xObjectSize, yObjectSize) / 2 : radius)\n      .attr('fill', intersection.includes(set) ? \"black\": 'rgb(233,233,233)')\n      .attr('stroke', \"black\")\n      .attr(\"in-intersection\", intersection.includes(set))\n    })\n\n\n\n  }\n\n  function intersectionTotals() {\n    var totals = {}\n    // intersectionValues.sort(function(a, b){ return intersectionKeySortingFunction(a, b) })\n\n    intersectionValues.map(function(k, i){ totals[k] = {'total': 0} })\n    cellKeys.map(function(k, i){\n      var e = elementExtractor(k, i);\n      if (totals[intersectionExtractor(k, i)]['total'] == 0) {\n        if (Array.isArray(e)) {\n          totals[intersectionExtractor(k, i)]['total']+= e.length\n          totals[intersectionExtractor(k, i)]['values'] = e\n        } else {\n          totals[intersectionExtractor(k, i)]['total']+= e\n        }\n\n      }\n    })\n    return totals\n  }\n\n  function setTotals(){\n    var totals = {}\n    // intersectionValues.sort(function(a, b){ return intersectionKeySortingFunction(a, b) })\n\n    setValues.map(function(k, i){ totals[k] = {'total': 0} })\n\n    cellKeys.map(function(k, i){\n      var e = elementExtractor(k, i);\n      if (Array.isArray(e)) {\n        totals[setExtractor(k, i)]['total']+= e.length\n      } else {\n        totals[setExtractor(k, i)]['total']+= e\n      }\n    })\n    return totals\n  }\n\n  upset.intersectionTotals = intersectionTotals\n  upset.setTotals = setTotals\n\n  return upset\n}\n","import {hypenate, safeSelect} from './helpers';\nexport function filterTable(selection) {\n  var\n  data,\n  namespace = 'd3sm-filter-table',\n  sortQ = false,\n  fieldFunction = function(record, columnKey, recordValue) {return recordValue}\n\n  filterTable.data = function(_) { return arguments.length ? (data = _, filterTable) : data}\n  filterTable.namespace = function(_) { return arguments.length ? (namespace = _, filterTable) : namespace}\n  filterTable.fieldFunction = function(_) { return arguments.length ? (fieldFunction = _, filterTable) : fieldFunction}\n\n  function filterTable () {\n\n    var\n    container = safeSelect(selection, 'div', 'filter-table').classed(hypenate(namespace,'container'),true),\n\n    inputGroup = safeSelect(container, 'div', 'filter-input-group').classed('input-group',true),\n      inputPrepend = safeSelect(inputGroup, 'div', 'input-group-prepend'),\n        inputPrependSpan = safeSelect(inputPrepend, 'span', 'input-group-text').classed('search-button', true),\n          inputPrependSpanIcon = safeSelect(inputPrependSpan,'i','fa fa-search'),\n\n      input = safeSelect(inputGroup, 'input', 'form-control').attr('placeholder', 'filter').attr('type', 'text'),\n      inputAppend = safeSelect(inputGroup, 'div', 'input-group-append'),\n        inputAppendButton = safeSelect(inputAppend, 'a', 'close-button').classed('btn btn-outline-secondary', true),\n          inputAppendButtonIcon = safeSelect(inputAppendButton, 'i', 'fa fa-close'),\n\n\n    closeButton = inputAppendButton,\n\n    rTable = safeSelect(container, 'div', 'table-responsive'),\n      table = safeSelect(rTable, 'table', 'table')\n      .classed('table-hover', true)\n      .classed('table-bordered', true)\n      .classed(\"table-striped\", true),\n        tHead = safeSelect(table, 'thead')\n        .classed(\"thead-dark\", true),\n          header = safeSelect(tHead, 'tr'),\n        tBody = safeSelect(table, 'tbody')\n\n\n    var\n    rows = d3.keys(data),\n    cols = d3.keys(data[rows[0]])\n\n    var th = makeColumns(header, cols)\n    var tr = makeRows(tBody, rows)\n    var tr = populateRecords(tr, cols)\n\n\n\n    input.on('input', function(d, i){\n      var\n      val = input.property('value'),\n      reg = new RegExp(val, 'gi'),\n      use\n      if (val == '') {use = rows} else {\n        use = []\n        rows.map(function(r, i){\n          var row = data[r]\n          var fieldJoin = cols.map(function(c, j){\n            return fieldFunction(row, c, row[c])\n            // return row[c]\n          }).join('')\n          var match = fieldJoin.match(reg)\n          if (match == null || match.join('') == '') {}\n          else { use.push(r) }\n        })\n      }\n\n      tr = makeRows(tBody, use)\n      tr = populateRecords(tr, cols)\n    })\n\n    closeButton.on('click', function(d, i){\n      input.property('value', '').dispatch('input')\n    })\n\n  }\n\n  function makeColumns(header, cols) {\n    var th = header.selectAll('th')\n    th = th.data(cols)\n    th.exit().remove()\n    th = th.merge(th.enter().append('th'))\n    th.attr('scope', 'col').text(function(d, i){return d})\n    return th\n  }\n\n  function makeRows(body, rows) {\n    var tr = body.selectAll('tr')\n    tr = tr.data(rows)\n    tr.exit().remove()\n    tr = tr.merge(tr.enter().append('tr'))\n    return tr\n  }\n\n  function populateRecords(rows, cols) {\n    rows.each(function(r, i){\n      var record = data[r],\n      t = d3.select(this)\n\n      var fields = t.selectAll('td')\n      fields = fields.data(cols)\n      fields.exit().remove()\n      fields = fields.merge(fields.enter().append('td'))\n\n      fields.attr('scope', function(f, j){ return j == 0 })\n      .html(function(f, j){ return fieldFunction(record, f, record[f]) })\n\n    })\n    return rows\n  }\n\n\n\n  return filterTable\n}\n"],"names":["uniqueElements","value","index","self","indexOf","getTranslation","transform","g","document","createElementNS","undefined","setAttributeNS","matrix","baseVal","consolidate","e","f","modifyHexidecimalColorLuminance","hex","lum","String","replace","length","c","i","rgb","parseInt","substr","Math","round","min","max","toString","quartiles","data","qKeys","q2","d3","median","lower","filter","x","upper","q1","q0","q3","q4","k0","k1","k2","k3","k4","obj","hypenate","Array","prototype","slice","call","arguments","join","number","precision","shift","reverseShift","numArray","split","getContainingSVG","element","parent","parentElement","tag","tagName","toLowerCase","safeSelect","sel","cls","clsStr","sSel","select","empty","append","classed","attr","tickRange","n","a","s","push","hasQ","array","item","includes","unique","flatten","flat","map","isArray","concat","whichBin","bins","j","consoleGroup","name","window","d3sm","debugQ","group","consoleGroupEnd","groupEnd","log","func","msg","table","setupContainer","selection","namespace","rect","fill","container","y","width","height","bgRect","defs","cpRect","raise","calculateWidthOfObject","freeSpace","numberOfObjects","minObjectWidth","maxObjectWidth","sizeOfSpacer","overflowQ","remainingSpace","objectWidth","calculateWidthOfSpacer","baseSpacerSize","spacersAtEachLevel","spacersNeededAtEachLevel","level","levelData","reduce","b","isNaN","whiskerPath","dir","w","h","per","o","hh","ww","p","groupingSpacer","scaleLinear","easeSin","cur","d","horizontalQ","outerWidth","current","currentNode","node","selectAll","transition","duration","transitionDuration","ease","easeFunc","remove","recursivelyPosition","currentSelection","enter","exit","merge","exitFunction","each","this","levelSpacer","spacerSize","move","currentElement","t","enterFunction","moveby","scale","toRemove","objectClass","objectSize","size","_","colorFunction","interpolateRgb","colors","k","v","category","interpolate","interpolation","domain","dataExtent","range","helperScale","match","key","type","hoverQ","opac","fillOpacity","strokeOpacity","colorBy","categories","modifyOpacity","valueExtractor","cat","categoryExtractor","tooltip","keys","values","header","on","mousemove","currentData","mouse","div","style","cardBody","tBody","text","tr","rowKey","rowIndex","bbox","getBoundingClientRect","innerWidth","scrollX","event","pageX","innerHeight","scrollY","pageY","selectFilter","selectionName","defaultValue","lastValue","selectAppendButton","inputGroup","input","inputAppendButton","options","closeButton","currentStyle","property","dispatch","use","val","reg","RegExp","option","currentOption","parseFloat","lasso","svg","chartContainer","chartOffset","objectsOffset","eventCatcher","xScale","path","line","yScale","curve","curveLinearClosed","instance","animationRate","opacity","dashArray","stroke","strokeWidth","lassoedStroke","lassoedStrokeWidth","easeExp","paths","pEnter","activeQ","objectContainer","allPoints","render","preventDefault","stopPropagation","addEventListener","drag","removeEventListener","currentPoints","applyPathAttributes","color","which","pt","invert","lastPt","p1","p2","sqrt","euclideanDistance","tickDistance","tick","detect","lassos","box","absolutePosition","left","top","right","bottom","boxPts","inAnyLassoQ","lassoPoints","every","polygonContains","coord","updateObjects","applyObjectAttributes","setQ","preLassoFill","preLassoStroke","preLassoStrokeWidth","lassoedFill","keyFrames","html","draw","toggle","state","thisSVG","absolute","elementPosition","svgPosition","relativePositionTo","axis","label","tickTickLabelSpacer","tickLabelMargin","reverseScaleQ","orient","verticalQ","bgcpRect","spaceX","spaceY","guideLinesQ","guidelineSpace","tickLabelMaxFontSize","backgroundFill","tickLabelTextAnchor","tickLabelRotation","tickData","categoricalQ","grouping","tickLabels","numberOfTicks","extent","tickValues","flatTickData","space","reverse","domainPadding","minObjectSize","maxObjectSize","objectSpacer","objClass","spacerFunction","moveXBy","moveYBy","mt","datum","dist","labelNameGroup","labelElement","that","tickLength","tickStroke","tickStrokeWidth","string","chars","roundTo","tickLabelFontSize","getComputedTextLength","labelHover","labelHoverOff","tickLabelOnClick","guideLineStroke","guideLineStrokeWidth","lineStroke","lineStrokeWidth","parentNode","tickLabelOnHoverFunc","gline","minorQ","ii","tickLabelMinFontSize","tickLabelFunc","bar","keyA","keyB","descending","CF","TTip","barPercent","barKeys","ordered","sort","sortingFunction","barValues","defaultExit","nodes","parentIndexArray","Number","fillColor","strokeColor","barStrokeWidth","bubbleHeatmap","xKey","yKey","rKey","vKey","xKeySortingFunction","xExtractor","yKeySortingFunction","yExtractor","bhm","cellKeys","rExtractor","vExtractor","xValues","yValues","xDim","yDim","rValues","ySize","xSize","ySpacer","ySpacerSize","xSpacer","xSpacerSize","cells","vValues","radius","scaled","bubbleStrokeWidth","heatmap","hm","yMinObjectSize","yMaxObjectSize","xMinObjectSize","xMaxObjectSize","lookup","positionedCellKeys","lookupValue","objectStrokeWidth","boxwhisker","quartilesKey","quartilesKeys","boxKeys","boxValues","whisk","uWhisk","lWhisk","quart","uQuart","lQuart","mQuart","boxStrokeWidth","r","dif","dd","whiskerWidthPercent","whiskerStrokeWidth","datatoggle","xAxisOptions","yAxisOptions","yAxisSelectQ","xAxisSelectQ","updateFunction","currentKeys","vals","filterSelects","dataopts","doEnter","sf","scatter","pointKeys","valueExtractorX","valueExtractorY","valueExtractorR","extentX","valuesX","domainPaddingX","extentY","valuesY","domainPaddingY","extentR","valuesR","domainPaddingR","minRadius","maxRadius","points","pExit","scaleX","scaleY","scaleR","pointStrokeWidth","plotZoom","chart","xAxis","yAxis","chartSel","xAxisSel","yAxisSel","setLocks","chartObjSel","chartObjTrans","cos","getBBox","xLock","yLock","zoom","chartBox","xAxisBox","yAxisBox","eventType","deltaY","wheelSpeed","shiftQ","shiftKey","applyX","applyY","xAxisObjSel","yAxisObjSel","reset","multiPlotZoom","xComponents","yComponents","xComponentsSel","yComponentsSel","xComponentObjSel","yComponentObjSel","violin","base","minMaxHexScale","scaledColor","mod","quartileKeys","pointsTooltip","violinKey","violinData","violinPointKey","violinPointData","calcValues","calculateViolinValues","violinPoints","violinPointsExtractor","violinPointsKeys","violinPointsValues","pk","violinPointValueExtractor","pointQuartiles","binned","histogram","frequencies","bin","minContourPoint","maxContourPoint","violinContourPoints","x0","x1","contour","pointValues","neededViolinValues","vk","violinKeys","frequencyMax","vScale","lArea","curveBasis","rArea","area","la","ra","quarts","pointsQ","ptsContainer","pts","ptsEnter","pMin","pMax","pointRadius","pointKey","random","pointColorFunc","createEvent","initEvent","dispatchEvent","tooltipKey","quartileKey","violinValues","numericLegend","fontSize","textColor","legend","linearGradient","m","categoricLegend","ascending","catKeys","cx","cy","lassoWidget","maxNumberOfGroups","dataExtractor","onError","submit","clickSend","removeBtnClickToRemove","tabs","panes","remainingTabs","lastTab","curText","populatePane","pane","lead","btns","cN","btn","lassoBtn","makeLassoButton","currentLasso","clearBtn","lassoBtnToggle","las","tableData","extracted","makeClearButton","monitorLassoButtonState","target","makeTable","headR","body","headerKeys","headerCols","updateTableHeaderColumns","bodyRows","updateTableRows","rowData","cols","splice","makeNewGroup","gs","newGroupNumber","li","insert","txt","insertTab","makeTabPane","card","tabList","tabContent","rightTabs","upset","setValues","intersectionValues","xObjectSize","circleStrokeWidth","setExtractor","intersectionExtractor","elementExtractor","elementValues","yObjectSpacer","setKeySortingFunction","intersectionKeySortingFunction","xObjectSpacer","yObjectSize","set","intersection","intersectionTotals","totals","total","setTotals","filterTable","sortQ","record","columnKey","recordValue","rows","th","makeColumns","populateRecords","makeRows","row","fieldFunction","fields","extractViolinValues","valueExtractorFunction","qKey","minPoint","maxPoint","interpolateColors","interpolateRgbBasis","truncateText","setupStandardChartContainers","margins","axes","leg","margin","pos","Height","svgSpace","margPx","chartSpace","axesSpace","legRect","drawingSpace","yAxisRect","plotRect","xAxisRect","myLog","warn","console","info","error","resizeDebounce","wait","resize","immediate","timeout","context","args","callNow","setTimeout","apply","debounce"],"mappings":"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"} diff --git a/build/js/d3sm.min.v0.0.4.js b/build/js/d3sm.min.v0.0.4.js deleted file mode 100644 index b4ca54b..0000000 --- a/build/js/d3sm.min.v0.0.4.js +++ /dev/null @@ -1,2 +0,0 @@ -var d3sm=function(t){"use strict";function e(t,e,n){return n.indexOf(t)===e}function n(t){var e=document.createElementNS("http://www.w3.org/2000/svg","g");t=void 0==t?"translate(0,0)":t,e.setAttributeNS(null,"transform",t);var n=e.transform.baseVal.consolidate().matrix;return[n.e,n.f]}function r(t,e){(t=String(t).replace(/[^0-9a-f]/gi,"")).length<6&&(t=t[0]+t[0]+t[1]+t[1]+t[2]+t[2]),e=e||0;var n,r,o="#";for(r=0;r<3;r++)n=parseInt(t.substr(2*r,2),16),o+=("00"+(n=Math.round(Math.min(Math.max(0,n+n*e),255)).toString(16))).substr(n.length);return o}function o(t,e){var n=d3.median(t),r=t.filter(function(t){return tn}),a=void 0==(a=d3.median(r))?n:a,i=void 0==(i=d3.min(r))?a:i,l=void 0==(l=d3.median(o))?n:l,c=void 0==(c=d3.max(o))?l:c,u="q0",s="q1",d="q2",f="q3",h="q4",g={};return void 0!=e&&5==e.length&&(u=e[0],s=e[1],d=e[2],f=e[3],h=e[4]),g[u]=i,g[s]=a,g[d]=n,g[f]=l,g[h]=c,g}function a(){return Array.prototype.slice.call(arguments).join("-")}function i(t,e){var n=function(t,e,n){n&&(e=-e);var r=(""+t).split("e");return+(r[0]+"e"+(r[1]?+r[1]+e:e))};return n(Math.round(n(t,e,!1)),e,!0)}function l(t){var e=t.parentElement,n=e.tagName.toLowerCase();return"svg"===n?e:"html"!==n?l(e):void 0}function c(t,e,n){var r=void 0==n?"":"."+n,o=t.select(e+r).empty()?t.append(e):t.select(e+r);return o.classed(r.replace(".",""),!0).attr("transform",void 0==o.attr("transform")?"translate(0,0)":o.attr("transform"))}function u(t,e,n){for(var r=[t],o=(e-t)/(n-1),a=0;a1?o:t*o),l=(i=i<0?0:i)/e;return a&&void 0!=n&&l=r.length?r.push(e.length-1):r[n]+=e.length-1;e.map(function(e,o){Array.isArray(e)&&t(e,n,r)});return r}(t);o=(e-n*r)/i.map(function(t,e){return 1*t/(e+1)}).reduce(function(t,e){return t+e},0);return isNaN(o)?0:o}function b(t,e,n,r,o,a,i){if("up"!=t&&"top"!=t&&1!=t||(t=!0),"down"!=t&&"bottom"!=t&&0!=t||(t=!1),i=void 0==i?"horizontal":i,a=void 0==a?1:a,"horizontal"!=i){var l=o*a,c=(r=t?r:-r,t?e+r:e),u=t?e:e+r,s=t?c:u;return f="M "+c+" "+o/2+" L "+u+" "+o/2+" M "+s+" "+(o/2-l/2)+" L "+s+" "+(o/2+l/2)+" "}var d=r*a,f="M "+r/2+" "+(c=t?n+o:n)+" L "+r/2+" "+(u=t?n:n+o)+" h "+-d/2+" 0 h "+d+" 0 ";return f}function k(){var t,e,n,r=!0,o=d3.scaleLinear(),a="category",i="d3sm-groupped-item",l=1e3,c=d3.easeSin,u="spacer",s=function(t){t.attr("transform",function(t,e){return"translate("+(r?window.outerWidth:0)+","+(r?0:window.outerWidth)+")"})},d=function(t){m("groupingSpacer","exiting with",{current:t,currentNode:t.node()}),t.selectAll("g").classed("to-remove",!0),t.transition().duration(.9*l).ease(c).attr("transform",function(t,e){return"translate("+(r?window.outerWidth:0)+","+(r?0:window.outerWidth)+")"}).remove()};function f(t,h,g){void 0==g&&(g=0);var p=t.selectAll("g."+u+'[level="'+g+'"]').data(h),m=p.enter().append("g").attr("level",g).attr("class",u),v=p.exit();p=p.merge(m),"function"==typeof d?v.each(function(t,e){d(d3.select(this))}):v.remove();var y=n/(g+1),x=0;return p.each(function(t,n){var h=d3.select(this);if(void 0==h.attr("transform")&&"function"==typeof s&&s(h),h.transition().duration(l).ease(c).attr("transform",function(t,e){return"translate("+(r?"scale"==a?o(t):x:0)+","+(r?0:"scale"==a?o(t):x)+")"}),Array.isArray(t)){x+=f(h,t,g+1);var m=h.selectAll("g."+u+'[level="'+g+'"] > g.'+i+"."+u);"function"==typeof d?m.each(function(t,e){d(d3.select(this))}):m.remove()}else{x+=e;var v=h.select("g."+u+'[level="'+g+'"] > g.'+i+"."+u);v.empty()&&(v=h.append("g").attr("class",i).classed(u,!0)),v.attr("parent-index",n);m=h.selectAll("g."+u+'[level="'+(g+1)+'"]');"function"==typeof d?m.each(function(t,e){d(d3.select(this))}):m.remove()}x+=n==p.size()-1?0:y}),x}return f.horizontalQ=function(t){return arguments.length?(r=t,f):r},f.scale=function(t){return arguments.length?(o=t,f):o},f.moveby=function(t){return arguments.length?(a=t,f):a},f.numberOfObjects=function(e){return arguments.length?(t=e,f):t},f.objectClass=function(t){return arguments.length?(i=t,f):i},f.objectSize=function(t){return arguments.length?(e=t,f):e},f.spacerSize=function(t){return arguments.length?(n=t,f):n},f.transitionDuration=function(t){return arguments.length?(l=t,f):l},f.easeFunc=function(t){return arguments.length?(c=t,f):c},f.namespace=function(t){return arguments.length?(u=t,f):u},f.enterFunction=function(t){return arguments.length?(s=t,f):s},f.exitFunction=function(t){return arguments.length?(d=t,f):d},f}var w=function(){return function(t,e){if(Array.isArray(t))return t;if(Symbol.iterator in Object(t))return function(t,e){var n=[],r=!0,o=!1,a=void 0;try{for(var i,l=t[Symbol.iterator]();!(r=(i=l.next()).done)&&(n.push(i.value),!e||n.length!==e);r=!0);}catch(t){o=!0,a=t}finally{try{!r&&l.return&&l.return()}finally{if(o)throw a}}return n}(t,e);throw new TypeError("Invalid attempt to destructure non-iterable instance")}}(),S=function(t){if(Array.isArray(t)){for(var e=0,n=Array(t.length);ewindow.innerWidth-window.scrollX&&(d=d3.event.pageX-b.width-15),f+b.height>window.innerHeight-window.scrollY&&(f=d3.event.pageY-b.height-15),"relative"==h.style("position")?h.style("position","absolute").style("left",d+"px").style("top",f+"px"):h.style("left",d+"px").style("top",f+"px"),h.attr("z-index",1e4)}return a.keys=function(t){return arguments.length?(e=t,a):e},a.values=function(t){return arguments.length?(n=t,a):n},a.header=function(t){return arguments.length?(r=t,a):r},a.data=function(t){return arguments.length?(o=t,a):o},a.selection=function(e){return arguments.length?(t=e,a):t},a}function A(t){var e,n="d3sm-select-filter",r="Select options:",o=void 0,i=void 0;function l(){var o=c(t,"div","input-group").classed(a(n,"container"),!0),l=(c(c(o,"div","select-prepend").classed("input-group-prepend",!0),"span","input-group-text").text(r),c(o,"select","custom-select").classed(a(n,"select"),!0)),s=c(c(o,"div","select-append").classed("input-group-prepend",!0),"a","filter-button").classed("btn btn-outline-secondary",!0),d=(c(s,"i","fa fa-filter"),c(o,"div","filter-input-group").classed("input-group",!0).classed("d-none",!0)),f=(c(c(c(d,"div","input-group-prepend"),"span","input-group-text").classed("search-button",!0),"i","fa fa-search"),c(d,"input","form-control").attr("placeholder","all").attr("type","text")),h=c(c(d,"div","input-group-append"),"a","close-button").classed("btn btn-outline-secondary",!0),g=(c(h,"i","fa fa-close"),d3.keys(e)),p=l.selectAll("option");p=(p=p.data(d3.keys(e))).merge(p.enter().append("option")).attr("value",function(t,e){return t}).text(function(t,e){return t});var m=h;s.on("click",function(t,e){var n=d.classed("d-none");d.classed("d-none",!n)}),m.on("click",function(t,e){f.property("value","").dispatch("input")}),f.on("input",function(t,n){var r,o=f.property("value"),a=new RegExp(o,"gi");""==o?r=g:(r=[],d3.keys(e).map(function(t,e){var n=t.match(a);null==n||""==n.join("")||r.push(t)})),(p=(p=l.selectAll("option")).data(r)).exit().remove(),p=p.merge(p.enter().append("option")).attr("value",function(t,e){return t}).text(function(t,e){return t});var c=u();i!=c&&(i=c,l.dispatch("change"))})}function u(){var n=t.select("select").property("value");return void 0==n||""==n?void 0==o?d3.keys(e)[0]:o:n}return l.data=function(t){return arguments.length?(e=t,l):e},l.namespace=function(t){return arguments.length?(n=t,l):n},l.selectionName=function(t){return arguments.length?(r=t,l):r},l.defaultValue=function(t){return arguments.length?(o=t,l):o},l.currentOption=u,l}function M(t){var e=t.attr("transform").split("translate("),n=w(e,2),r=(n[0],n[1].split(",")),o=w(r,2),a=o[0],i=o[1];return i.split(")"),[parseFloat(a),parseFloat(i)]}function E(t){var e,n,r,o,i,l,u,s,f,h,g="d3sm-lasso",p=!1,m=[],v=[],y=d3.line().x(function(t,e){return void 0!=s?s(t[0]):t[0]}).y(function(t,e){return void 0!=f?f(t[1]):t[1]}).curve(d3.curveLinearClosed),x=0,b=10,k="#17a2b8",w="10s",S=.3,j="5, 10",z="black",A=2,E="white",O="black",L=3,C=1e3,F=d3.easeExp;function Q(){var t,e;p&&((t=(t=c(n,"g","lasso-container").selectAll('path[instance="'+x+'"]')).data(v)).exit().remove(),e=t.enter().append("path"),V(t=t.merge(e).transition().duration(C).ease(F)))}function B(){var t=c(n,"g","lasso-container");t.selectAll('path[instance="'+x+'"]').remove();t.remove(),n.selectAll(r).classed("in-lasso",!1),Y()}function q(t){t.preventDefault(),t.stopPropagation();var r=c(n,"g","lasso-container");m=[],e.node().addEventListener("mousemove",D),e.node().addEventListener("mouseup",function(t){e.node().removeEventListener("mousemove",D),v.push(m),v=d(v)}),V(h=r.append("path").data([m]))}function V(t){t.attr("class",a(g,"lasso-path")).style("opacity",S).attr("fill",k).attr("d",y).attr("instance",x).style("stroke-dasharray",j).attr("stroke",z).attr("stroke-width",A).style("animation","lassoDash "+w+" linear").style("animation-iteration-count","infinite")}function D(t){if(void 0!=u&&u.dispatch(a(g,"drag")),1==t.which){d3.event=t;var r=d3.mouse(n.node());r=d3.mouse(e.node());if(void 0!=s&&(r[0]=s.invert(r[0])),void 0!=f&&(r[1]=f.invert(r[1])),r[0]=r[0]-i[0]-l[0],r[1]=r[1]-i[1]-l[1],m.length){var o=m[m.length-1],c=[r[0],r[1]],d=[o[0],o[1]];s&&(d[0]=s(d[0]),c[0]=s(c[0])),f&&(d[1]=f(d[1]),c[1]=f(c[1])),function(t,e){var n=t[0]-e[0],r=t[1]-e[1];return Math.sqrt(n*n+r*r)}(d,c)>b&&_(r)}else _(r)}}function _(t){if(void 0!=t){if(m.push(t),h.attr("d",y),m.length<3)return;K(v.concat([m]))}else K(v)}function K(t){return void 0==t&&(t=v),n.selectAll(r).each(function(e,n){var r=d3.select(this),o=r.absolutePosition(),a=[[o.left-i[0]-l[0],o.top-i[1]-l[1]],[o.right-i[0]-l[0],o.top-i[1]-l[1]],[o.left-i[0]-l[0],o.bottom-i[1]-l[1]],[o.right-i[0]-l[0],o.bottom-i[1]-l[1]]];void 0!=s&&(a[0][0]=s.invert(a[0][0]),a[1][0]=s.invert(a[1][0]),a[2][0]=s.invert(a[2][0]),a[3][0]=s.invert(a[3][0])),void 0!=f&&(a[0][1]=f.invert(a[0][1]),a[1][1]=f.invert(a[1][1]),a[2][1]=f.invert(a[2][1]),a[3][1]=f.invert(a[3][1]));var c=!1;for(n=0;nwindow.innerWidth&&o.style("left",d3.event.pageX-15-300+"px")}function ot(t,e){var n=d3.select(this).style("fill","black");if(d3.select(n.node().parentNode).select("line."+a(B,"tick")).attr("stroke",K).attr("stroke-width",Y),M){var o=d3.select(n.node().parentNode).select("line."+a(B,"guideline")),i=o.attr("minor");o.attr("stroke",function(t,e){return"true"==i?r(I,.8):I}).attr("stroke-width",function(t,e){return"true"==i?.8*J:J})}d3.select("#"+a(B,"guideline-tooltip")).remove()}return nt.label=function(t){return arguments.length?(m=t,nt):m},nt.tickTickLabelSpacer=function(t){return arguments.length?(X=t,nt):X},nt.tickLabelMargin=function(t){return arguments.length?(R=t,nt):R},nt.selection=function(e){return arguments.length?(t=e,nt):t},nt.orient=function(t){return arguments.length?(b=t,nt):b},nt.spaceX=function(t){return arguments.length?(w=t,nt):w},nt.spaceY=function(t){return arguments.length?(j=t,nt):j},nt.overflowQ=function(t){return arguments.length?(z=t,nt):z},nt.categoricalQ=function(t){return arguments.length?(A=t,nt):A},nt.guideLinesQ=function(t){return arguments.length?(M=t,nt):M},nt.grouping=function(t){return arguments.length?(e=t,nt):e},nt.scale=function(t){return arguments.length?(E=t,nt):E},nt.domainPadding=function(t){return arguments.length?(O=t,nt):O},nt.objectSpacer=function(t){return arguments.length?(L=t,nt):L},nt.minObjectSize=function(t){return arguments.length?(C=t,nt):C},nt.maxObjectSize=function(t){return arguments.length?(F=t,nt):F},nt.namespace=function(t){return arguments.length?(B=t,nt):B},nt.backgroundFill=function(t){return arguments.length?(Q=t,nt):Q},nt.objectClass=function(t){return arguments.length?(q=t,nt):q},nt.tickLabels=function(t){return arguments.length?(n=t,nt):n},nt.tickValues=function(t){return arguments.length?(o=t,nt):o},nt.numberOfTicks=function(t){return arguments.length?(V=t,nt):V},nt.lineStroke=function(t){return arguments.length?(D=t,nt):D},nt.lineStrokeWidth=function(t){return arguments.length?(_=t,nt):_},nt.tickStroke=function(t){return arguments.length?(K=t,nt):K},nt.tickStrokeWidth=function(t){return arguments.length?(Y=t,nt):Y},nt.tickLength=function(t){return arguments.length?(P=t,nt):P},nt.tickLabelFontSize=function(t){return arguments.length?(W=t,nt):W},nt.tickLabelMinFontSize=function(t){return arguments.length?(N=t,nt):N},nt.tickLabelMaxFontSize=function(t){return arguments.length?(T=t,nt):T},nt.tickLabelTextAnchor=function(t){return arguments.length?(l=t,nt):l},nt.tickLabelRotation=function(t){return arguments.length?(d=t,nt):d},nt.tickLabelFunc=function(t){return arguments.length?(G=t,nt):G},nt.tickLabelOnClick=function(t){return arguments.length?(H=t,nt):H},nt.guidelineSpace=function(t){return arguments.length?(h=t,nt):h},nt.guideLineStroke=function(t){return arguments.length?(I=t,nt):I},nt.guideLineStrokeWidth=function(t){return arguments.length?(J=t,nt):J},nt.transitionDuration=function(t){return arguments.length?(U=t,nt):U},nt.easeFunc=function(t){return arguments.length?($=t,nt):$},nt.objectSize=function(t){return arguments.length?(g=t,nt):g},nt.spacerSize=function(t){return arguments.length?(p=t,nt):p},nt.roundTo=function(t){return arguments.length?(tt=t,nt):tt},nt.reverseScaleQ=function(t){return arguments.length?(et=t,nt):et},nt.tickLabelOnHoverFunc=function(t){return arguments.length?(Z=t,nt):Z},nt},O.bar=function(t){var e,n,r,o,a,i,l,u,s="horizontal",d=!1,h=function(t,n){return e[t]},g=function(t,n){return d3.descending(e[t],e[n])},p=d3.scaleLinear(),m=.5,b=.05,w=50,A=100,M=2,E=j(),O="transparent",L="d3sm-bar",C="bar",F=1e3,Q=d3.easeExp,B=z(),q=1;function V(){var j="horizontal"==s||"bottom"==s||"top"==s,z=!j,V=v(t,L,{x:0,y:0,width:n,height:r},O);a=d3.keys(e),i=a.map(h);var D=void 0==o?a.sort(g):o,_=(a=f(D)).length,K=[Math.min.apply(Math,S(i))-m,Math.max.apply(Math,S(i))+m];p.domain(K).range(j?[0,r]:"right"==s?[0,n]:[n,0]);var Y=j?n:r;l=void 0==l?y(Y,_,w,A,b,d):l,u=void 0==u?x(a,Y,l,_,b,d):u;var P=k().horizontalQ(j).scale(p).moveby("category").numberOfObjects(_).objectClass(C).objectSize(l).spacerSize(u).transitionDuration(F).easeFunc(Q).namespace(L),X=P.exitFunction();P.exitFunction(function(t){void 0==l&&console.log(t.nodes(),l),X(t),t.selectAll("g").classed("to-remove",!0),t.selectAll("* > rect").transition().duration(F).attr("transform",function(t,e){return"translate(0,"+(z?0:p(K[1]))+")"}).attr("width",j?l:0).attr("height",z?l:0).remove()}),P(V,D,0);var R=[];V.selectAll("g:not(.to-remove)."+C).each(function(t,e){R.push(Number(d3.select(this).attr("parent-index")))}),E="index"==E.colorBy()?E.dataExtent([0,Math.max.apply(Math,R)]):E.dataExtent(K),V.selectAll("g."+C+":not(.to-remove)").each(function(t,n){var r=d3.select(this),o=(e[t],h(t,n)),a=(n=void 0==r.attr("parent-index")?n:r.attr("parent-index"),E(t,o,n,"fill")),i=E(t,o,n,"stroke"),u=c(r,"rect","bar-rect");void 0==u.attr("transform")&&u.attr("transform",function(t,e){return"translate(0,"+(z?0:p(K[1]))+")"}).attr("width",j?l:0).attr("height",z?l:0),u.transition().duration(F).ease(Q).attr("transform",function(t,e){return"translate("+(j?l-l*q:"right"==s?p(K[1])-p(o):l-l*q)+","+(z?l-l*q:p(K[1])-p(o))+")"}).attr("width",j?l*q:p(o)).attr("height",z?l*q:p(o)).attr("fill",a).attr("stroke",i).attr("stroke-width",M),r.on("mouseover",function(t,e){V.selectAll("g."+C).style("opacity",.2),r.style("opacity",1),u.attr("stroke-width",2*M)}),r.on("mouseout",function(){V.selectAll("g."+C).style("opacity",1),u.attr("stroke-width",M)})}),B.selection(V.selectAll(".bar-rect")).data(e),B()}return V.selection=function(e){return arguments.length?(t=e,V):t},V.data=function(t){return arguments.length?(e=t,V):e},V.orient=function(t){return arguments.length?(s=t,V):s},V.spaceX=function(t){return arguments.length?(n=t,V):n},V.spaceY=function(t){return arguments.length?(r=t,V):r},V.overflowQ=function(t){return arguments.length?(d=t,V):d},V.grouping=function(t){return arguments.length?(o=t,V):o},V.valueExtractor=function(t){return arguments.length?(h=t,V):h},V.sortingFunction=function(t){return arguments.length?(g=t,V):g},V.scale=function(t){return arguments.length?(p=t,V):p},V.domainPadding=function(t){return arguments.length?(m=t,V):m},V.objectSpacer=function(t){return arguments.length?(b=t,V):b},V.minObjectSize=function(t){return arguments.length?(w=t,V):w},V.maxObjectSize=function(t){return arguments.length?(A=t,V):A},V.barStrokeWidth=function(t){return arguments.length?(M=t,V):M},V.colorFunction=function(t){return arguments.length?(E=t,V):E},V.backgroundFill=function(t){return arguments.length?(O=t,V):O},V.namespace=function(t){return arguments.length?(L=t,V):L},V.objectClass=function(t){return arguments.length?(C=t,V):C},V.transitionDuration=function(t){return arguments.length?(F=t,V):F},V.easeFunc=function(t){return arguments.length?(Q=t,V):Q},V.barKeys=function(t){return arguments.length?(a=t,V):a},V.barValues=function(t){return arguments.length?(i=t,V):i},V.objectSize=function(t){return arguments.length?(l=t,V):l},V.spacerSize=function(t){return arguments.length?(u=t,V):u},V.tooltip=function(t){return arguments.length?(B=t,V):B},V.barPercent=function(t){return arguments.length?(q=t,V):q},V},O.bubbleHeatmap=function(t){var e,n,r,o,i,l,u,s,f,h,g,p,b="x",w="y",A="r",M="v",E=function(t,n){return e[t][b]},O=function(t,n){return e[t][w]},L=function(t,n){return e[t][A]},C=function(t,n){return e[t][M]},F=!1,Q=d3.scaleLinear(),B=.5,q=0,V=50,D=100,_=2,K="transparent",Y="d3sm-bubble",P="bubble",X=1e3,R=d3.easeExp,W=function(t,e){return E(t)-E(e)},N=function(t,e){return O(t)-O(e)},T=j().colorBy("value"),G=z();function H(){var b=v(t,Y,{x:0,y:0,width:n,height:r},K);(o=d3.keys(e)).sort(function(t,e){return W(t,e)||N(t,e)}),m("bubbleHeatmap","cells are sorted by",o),i=d(o.map(E)),l=d(o.map(O)),u=d(o.map(L)),s=d(o.map(C)),m("bubbleHeatmap","x and y keys are",{x:i,y:l});var w=i.length,j=l.length,z=[Math.min.apply(Math,S(u))-B,Math.max.apply(Math,S(u))+B];g=y(r,j,V,D,q,F),f=y(n,w,V,D,q,F),p=x(l,r,g,j,q,F),h=x(i,n,f,w,q,F),m("bubbleHeatmap","size of",{x:f,y:g}),Q.domain(z).range([Math.min(V/2,Math.min(g,f)/2),Math.min(g,f)/2]);var A=k().horizontalQ(!1).moveby("category").numberOfObjects(j).objectClass(a(P,"row")).objectSize(g).spacerSize(p).transitionDuration(X).easeFunc(R).namespace("row"),M=k().horizontalQ(!0).moveby("category").numberOfObjects(w).objectClass(P).objectSize(f).spacerSize(h).transitionDuration(X).easeFunc(R);A(b,l,0),b.selectAll("g."+a(P,"row")).each(function(t,e){M(d3.select(this),i,0)});var H=b.selectAll("g:not(.to-remove)."+P).data(o),Z=[];H.each(function(t,e){Z.push(Number(d3.select(this).attr("parent-index")))}),T="index"==T.colorBy()?T.dataExtent([0,Math.max.apply(Math,Z)]):T.dataExtent([0,Math.max.apply(Math,S(s))]),H.each(function(t,n){m("bubbleHeatmap","each cell",{key:t,index:n,node:d3.select(this).node()});var r=d3.select(this),o=(e[t],C(t,n)),i=L(t,n),l=(n=void 0==r.attr("parent-index")?n:r.attr("parent-index"),T(t,o,n,"fill")),u=T(t,o,n,"stroke");m("bubbleHeatmap","radius",{radius:i,scaled:Q(i),extent:z,range:Q.range()}),c(r,"circle",a(P,"circle")).attr("cx",f/2).attr("cy",g/2).attr("r",Q(i)).attr("fill",l).attr("stroke",u).attr("stroke-width",_)}),G.selection(H.selectAll("circle."+a(P,"circle"))).data(e),G()}return H.selection=function(e){return arguments.length?(t=e,H):t},H.data=function(t){return arguments.length?(e=t,H):e},H.spaceX=function(t){return arguments.length?(n=t,H):n},H.spaceY=function(t){return arguments.length?(r=t,H):r},H.xKey=function(t){return arguments.length?(b=t,H):b},H.yKey=function(t){return arguments.length?(w=t,H):w},H.rKey=function(t){return arguments.length?(A=t,H):A},H.vKey=function(t){return arguments.length?(M=t,H):M},H.cellKeys=function(t){return arguments.length?(o=t,H):o},H.xValues=function(t){return arguments.length?(i=t,H):i},H.yValues=function(t){return arguments.length?(l=t,H):l},H.rValues=function(t){return arguments.length?(u=t,H):u},H.vValues=function(t){return arguments.length?(s=t,H):s},H.xExtractor=function(t){return arguments.length?(E=t,H):E},H.yExtractor=function(t){return arguments.length?(O=t,H):O},H.rExtractor=function(t){return arguments.length?(L=t,H):L},H.vExtractor=function(t){return arguments.length?(C=t,H):C},H.overflowQ=function(t){return arguments.length?(F=t,H):F},H.scale=function(t){return arguments.length?(Q=t,H):Q},H.domainPadding=function(t){return arguments.length?(B=t,H):B},H.objectSpacer=function(t){return arguments.length?q=t:e},H.minObjectSize=function(t){return arguments.length?(V=t,H):V},H.maxObjectSize=function(t){return arguments.length?(D=t,H):D},H.bubbleStrokeWidth=function(t){return arguments.length?(_=t,H):_},H.backgroundFill=function(t){return arguments.length?(K=t,H):K},H.namespace=function(t){return arguments.length?(Y=t,H):Y},H.objectClass=function(t){return arguments.length?(P=t,H):P},H.transitionDuration=function(t){return arguments.length?(X=t,H):X},H.easeFunc=function(t){return arguments.length?(R=t,H):R},H.tooltip=function(t){return arguments.length?(G=t,H):G},H.colorFunction=function(t){return arguments.length?(T=t,H):T},H.xSize=function(t){return arguments.length?(f=t,H):f},H.xSpacerSize=function(t){return arguments.length?(h=t,H):h},H.ySize=function(t){return arguments.length?(g=t,H):g},H.ySpacerSize=function(t){return arguments.length?(p=t,H):p},H},O.heatmap=function(t){var e,n,r,o,i,l,u,s,f,h,g,p="x",b="y",w="v",A=function(t,n){return e[t][p]},M=function(t,n){return e[t][b]},E=function(t,n){return e[t][w]},O=!1,L=0,C=50,F=50,Q=100,B=100,q=2,V="transparent",D="d3sm-heatmap",_="heatmap",K=1e3,Y=d3.easeExp,P=function(t,e){return i.indexOf(A(t))-i.indexOf(A(e))},X=function(t,e){return l.indexOf(M(t))-l.indexOf(M(e))},R=j().colorBy("category"),W=z();function N(){var p=v(t,D,{x:0,y:0,width:n,height:r},V);o=d3.keys(e),i=d(o.map(A)),l=d(o.map(M)),u=d(o.map(E)),o.sort(function(t,e){return P(t,e)||X(t,e)}),m("heatmap","cells are sorted by",o),m("heatmap","x and y keys are",{x:i,y:l});var b=i.length,w=l.length;h=y(r,w,C,B,L,O),s=y(n,b,F,Q,L,O),g=x(l,r,h,w,L,O),f=x(i,n,s,b,L,O),m("heatmap","size of",{x:s,y:h});var j=k().horizontalQ(!1).moveby("category").numberOfObjects(w).objectClass(a(_,"row")).objectSize(h+g).spacerSize(0).transitionDuration(K).easeFunc(Y).namespace("row"),z=k().horizontalQ(!0).moveby("category").numberOfObjects(b).objectClass(_).objectSize(s+f).spacerSize(0).transitionDuration(K).easeFunc(Y);j(p,l,0),p.selectAll("g."+a(_,"row")).each(function(t,e){z(d3.select(this),i,0)});var N=p.selectAll("g:not(.to-remove)."+_);if(o.length!=l.length*i.length){var T={};o.map(function(t,e){T[A(t)+"::"+M(t)]=t});for(var G=[],H=0;Hr?r:n}).attr("fill",h).attr("stroke",v).attr("stroke-width",Q).attr("transform",function(t,e){return"translate("+(g?l/2:A(u))+","+(j?l/2:A(R[1])-A(u))+")"}),k.transition().duration(_).ease(K).attr("d",function(t,e){var n=g?A(i)-A(a):l;return b(!1,0,0,j?A(i)-A(a):l,n,C,d)}).attr("transform",function(t,e){return"translate("+(g?0:A(i))+","+(j?0:A(R[1])-A(i))+")"}).attr("stroke","black").attr("stroke-width",B).attr("fill","none"),x.transition().duration(_).ease(K).attr("d",function(t,e){var n=g?A(f)-A(s):l;return b(!0,0,0,j?A(f)-A(s):l,n,C,d)}).attr("transform",function(t,e){return"translate("+(g?0:A(s))+","+(j?0:A(R[1])-A(f))+")"}).attr("stroke","black").attr("stroke-width",B).attr("fill","none")}),Y.selection(z.selectAll("g:not(.to-remove)."+D)).data(e),Y()}return P.selection=function(e){return arguments.length?(t=e,P):t},P.data=function(t){return arguments.length?(e=t,P):e},P.orient=function(t){return arguments.length?(d=t,P):d},P.spaceX=function(t){return arguments.length?(n=t,P):n},P.spaceY=function(t){return arguments.length?(r=t,P):r},P.overflowQ=function(t){return arguments.length?(h=t,P):h},P.grouping=function(t){return arguments.length?(o=t,P):o},P.quartilesKey=function(t){return arguments.length?(g=t,P):g},P.quartilesKeys=function(t){return arguments.length?(p=t,P):p},P.valueExtractor=function(t){return arguments.length?(m=t,P):m},P.sortingFunction=function(t){return arguments.length?(w=t,P):w},P.scale=function(t){return arguments.length?(A=t,P):A},P.domainPadding=function(t){return arguments.length?(M=t,P):M},P.objectSpacer=function(t){return arguments.length?(E=t,P):E},P.minObjectSize=function(t){return arguments.length?(O=t,P):O},P.maxObjectSize=function(t){return arguments.length?(L=t,P):L},P.whiskerWidthPercent=function(t){return arguments.length?(C=t,P):C},P.colorFunction=function(t){return arguments.length?(F=t,P):F},P.boxStrokeWidth=function(t){return arguments.length?(Q=t,P):Q},P.whiskerStrokeWidth=function(t){return arguments.length?(B=t,P):B},P.backgroundFill=function(t){return arguments.length?(q=t,P):q},P.namespace=function(t){return arguments.length?(V=t,P):V},P.objectClass=function(t){return arguments.length?(D=t,P):D},P.transitionDuration=function(t){return arguments.length?(_=t,P):_},P.easeFunc=function(t){return arguments.length?(K=t,P):K},P.boxKeys=function(t){return arguments.length?(a=t,P):a},P.boxValues=function(t){return arguments.length?(i=t,P):i},P.objectSize=function(t){return arguments.length?(l=t,P):l},P.spacerSize=function(t){return arguments.length?(u=t,P):u},P.tooltip=function(t){return arguments.length?(Y=t,P):Y},P},O.colorFunction=j,O.datatoggle=function(t){var e,n,r,o=function(){},i="d3sm-databar",l=!1,u=!1;d.xAxisSelectQ=function(t){return arguments.length?(l=t,d):l},d.yAxisSelectQ=function(t){return arguments.length?(u=t,d):u},d.xAxisOptions=function(t){return arguments.length?(e=t,d):e},d.yAxisOptions=function(t){return arguments.length?(n=t,d):n},d.data=function(t){return arguments.length?(r=t,d):r},d.updateFunction=function(t){return arguments.length?(o=t,d):o},d.namespace=function(t){return arguments.length?(i=t,d):i},d.currentKeys=function(){var t={};return d3.keys(s).map(function(e,n){t[e]=s[e].currentOption()}),t};var s={};function d(){var e=c(t,"div","d-inline-flex flex-row flex-wrap").selectAll("div."+a(i,"select-filter"));e.exit().remove();var n=(e=e.data(d3.keys(r))).enter().append("div").attr("class","select-filter");return(e=e.merge(n).style("margin-right","10px")).each(function(t,e){var n=A(d3.select(this)).data(r[t]).namespace(a(i,t)).selectionName(t);n(),s[t]=n}),t.selectAll("select").on("change",function(){o()}),d}return d},O.groupingSpacer=k,O.tooltip=z,O.scatter=function(t){var e,n,r,o,a,i,l,c=d3.scaleLinear(),u=.5,s=function(t,n){return e[t].x},d=d3.scaleLinear(),f=.5,h=function(t,n){return e[t].y},g=d3.scaleLinear(),p=.5,m=function(t,e){return 2},y=2,x=10,b=2,k=j(),w="transparent",A="d3sm-scatter",M="scatter-point",E=1e3,O=d3.easeExp,L=z();function C(){var j=v(t,A,{x:0,y:0,width:n,height:r},w);o=d3.keys(e),a=o.map(s),i=o.map(h),l=o.map(m),o.length;var z=[Math.min.apply(Math,S(a))-u,Math.max.apply(Math,S(a))+u],C=[Math.min.apply(Math,S(i))-f,Math.max.apply(Math,S(i))+f],F=[Math.min.apply(Math,S(l))-p,Math.max.apply(Math,S(l))+p];c.domain(z).range([0,n]),d.domain(C).range([r,0]),g.domain(F).range([y,x]);var Q=j.selectAll("."+M),B=(Q=Q.data(o)).enter().append("circle").attr("class",M).attr("cx",0).attr("cy",r).attr("r",0),q=Q.exit();(Q=Q.merge(B)).each(function(t,n){var r=d3.select(this),o=e[t],u=a[n],s=i[n],f=l[n],h=k(t,o,n,"fill"),p=k(t,o,n,"stroke");r.transition().duration(E).ease(O).attr("cx",c(u)).attr("cy",d(s)).attr("r",g(f)).attr("fill",h).attr("stroke",p).attr("stroke-width",b),r.on("mouseover",function(t,e){Q.style("opacity",.2),r.style("opacity",1),r.transition().duration(E/2).ease(O).attr("stroke-width",2*b).attr("r",1.5*g(f))}),r.node().addEventListener("mouseout",function(){j.selectAll("."+M).style("opacity",1),r.transition().duration(E/2).ease(O).attr("stroke-width",b).attr("r",g(f))})}),q.transition().duration(E).ease(O).attr("cx",0).attr("cy",r).attr("r",0).remove(),L.selection(Q).data(e),L()}return C.selection=function(e){return arguments.length?(t=e,C):t},C.data=function(t){return arguments.length?(e=t,C):e},C.spaceX=function(t){return arguments.length?(n=t,C):n},C.spaceY=function(t){return arguments.length?(r=t,C):r},C.scaleX=function(t){return arguments.length?(c=t,C):c},C.domainPaddingX=function(t){return arguments.length?(u=t,C):u},C.valueExtractorX=function(t){return arguments.length?(s=t,C):s},C.scaleY=function(t){return arguments.length?(d=t,C):d},C.domainPaddingY=function(t){return arguments.length?(f=t,C):f},C.valueExtractorY=function(t){return arguments.length?(h=t,C):h},C.scaleR=function(t){return arguments.length?(g=t,C):g},C.domainPaddingR=function(t){return arguments.length?(p=t,C):p},C.valueExtractorR=function(t){return arguments.length?(m=t,C):m},C.minRadius=function(t){return arguments.length?(y=t,C):y},C.maxRadius=function(t){return arguments.length?(x=t,C):x},C.pointStrokeWidth=function(t){return arguments.length?(b=t,C):b},C.colorFunction=function(t){return arguments.length?(k=t,C):k},C.backgroundFill=function(t){return arguments.length?(w=t,C):w},C.namespace=function(t){return arguments.length?(A=t,C):A},C.objectClass=function(t){return arguments.length?(M=t,C):M},C.transitionDuration=function(t){return arguments.length?(E=t,C):E},C.easeFunc=function(t){return arguments.length?(O=t,C):O},C.pointKeys=function(t){return arguments.length?(o=t,C):o},C.valuesX=function(t){return arguments.length?(a=t,C):a},C.valuesY=function(t){return arguments.length?(i=t,C):i},C.valuesR=function(t){return arguments.length?(l=t,C):l},C.tooltip=function(t){return arguments.length?(L=t,C):L},C},O.plotZoom=function(t,e,r){var o,i=20,l=void 0==t.orient?"horizontal":t.orient(),c=t.spaceX(),u=t.spaceY(),s=t.selection(),d=e.selection(),f=r.selection();function h(){var e=s.select("."+a(t.namespace(),"object-container")),r=n(e.attr("transform")),o=e.attr("transform","translate(0,0)");c=s.node().getBBox().width-.9*t.spaceX(),u=s.node().getBBox().height-.9*t.spaceY(),o.attr("transform","translate("+r[0]+","+r[1]+")"),m("plotZoom","setLocks",{xLock:c,yLock:u})}function g(){var g,p;h(),"2D"==l&&(g=!0,p=!0),"horizontal"==l&&(g=!0,p=!1),"vertical"==l&&(p=!0,g=!1);var m=d3.event.transform,v=s.node().getBBox(),y=d.node().getBBox(),x=d.node().getBBox();if(v.width,v.x,v.height,v.y,y.width,y.height,x.width,x.height,"wheel"==o){d3.event.preventDefault();var b=d3.event.deltaY*i,k=d3.event.shiftKey;(m="2D"==l?k?{k:1,x:b,y:0}:{k:1,x:0,y:b}:g?{k:1,x:b,y:0}:{k:1,x:0,y:b}).applyX=function(t){return t*this.k+-1*this.x},m.applyY=function(t){return t*this.k+-1*this.y}}var w=s.select("."+a(t.namespace(),"object-container")),S=d.select("."+a(e.namespace(),"object-container")),j=f.select("."+a(r.namespace(),"object-container")),z=n(w.attr("transform")),A=(n(S.attr("transform")),n(j.attr("transform")),g?m.applyX(z[0]):0);g&&(A=A<-c?(m.x=0,-c):(m.x=0,Math.min(A,0)));var M=p?m.applyY(z[1]):0;p&&(M=M<-u?(m.y=0,-u):(m.y=0,Math.min(M,0))),w.attr("transform","translate("+A+","+M+")"),g&&S.attr("transform","translate("+A+",0)"),p&&j.attr("transform","translate(0,"+M+")")}return d3.select(s.thisSVG()),g.eventType=function(t){return arguments.length?(o=t,g):o},g.wheelSpeed=function(t){return arguments.length?(i=t,g):i},g.orient=function(t){return arguments.length?(l=t,g):l},g.xLock=function(t){return arguments.length?(c=t,g):c},g.yLock=function(t){return arguments.length?(u=t,g):u},g.setLocks=h,g.reset=function(){var n=s.select("."+a(t.namespace(),"object-container")),o=d.select("."+a(e.namespace(),"object-container")),i=f.select("."+a(r.namespace(),"object-container"));n.attr("transform","translate(0,0)"),o.attr("transform","translate(0,0)"),i.attr("transform","translate(0,0)")},g},O.multiPlotZoom=function(t){var e,r=20,o=void 0==t.orient?"horizontal":t.orient(),i=t.spaceX(),l=t.spaceY(),c=t.selection(),u=(d3.select(c.thisSVG()),[]),s=[];function d(){var e=c.select("."+a(t.namespace(),"object-container")),r=n(e.attr("transform")),o=e.attr("transform","translate(0,0)");i=c.node().getBBox().width-t.spaceX(),l=c.node().getBBox().height-t.spaceY(),o.attr("transform","translate("+r[0]+","+r[1]+")"),m("plotZoom","setLocks",{xLock:i,yLock:l})}function f(){d();var f,h,g=u.map(function(t,e){return t.selection()}),p=s.map(function(t,e){return t.selection()});"2D"==o&&(f=!0,h=!0),"horizontal"==o&&(f=!0,h=!1),"vertical"==o&&(h=!0,f=!1);var m=d3.event.transform,v=c.node().getBBox();if(g.map(function(t,e){return t.node().getBBox()}),g.map(function(t,e){return t.node().getBBox()}),v.width,v.x,v.height,v.y,"wheel"==e){d3.event.preventDefault();var y=d3.event.deltaY*r,x=d3.event.shiftKey;(m="2D"==o?x?{k:1,x:y,y:0}:{k:1,x:0,y:y}:f?{k:1,x:y,y:0}:{k:1,x:0,y:y}).applyX=function(t){return t*this.k+-1*this.x},m.applyY=function(t){return t*this.k+-1*this.y}}var b=c.select("."+a(t.namespace(),"object-container")),k=g.map(function(t,e){return t.select("."+a(u[e].namespace(),"object-container"))}),w=p.map(function(t,e){return t.select("."+a(s[e].namespace(),"object-container"))}),S=n(b.attr("transform")),j=(k.map(function(t,e){return n(t.attr("transform"))}),w.map(function(t,e){return n(t.attr("transform"))}),f?m.applyX(S[0]):0);f&&(j=j<-i?(m.x=0,-i):(m.x=0,Math.min(j,0)));var z=h?m.applyY(S[1]):0;h&&(z=z<-l?(m.y=0,-l):(m.y=0,Math.min(z,0))),b.attr("transform","translate("+j+","+z+")"),f&&k.map(function(t,e){t.attr("transform","translate("+j+",0)")}),h&&w.map(function(t,e){t.attr("transform","translate(0,"+z+")")})}return f.eventType=function(t){return arguments.length?(e=t,f):e},f.wheelSpeed=function(t){return arguments.length?(r=t,f):r},f.orient=function(t){return arguments.length?(o=t,f):o},f.xLock=function(t){return arguments.length?(i=t,f):i},f.yLock=function(t){return arguments.length?(l=t,f):l},f.xComponents=function(t){return arguments.length?(u=t,f):u},f.yComponents=function(t){return arguments.length?(s=t,f):s},f.setLocks=d,f.reset=function(){var e=c.select("."+a(t.namespace(),"object-container")),n=xAxisSel.select("."+a(xAxis.namespace(),"object-container")),r=yAxisSel.select("."+a(yAxis.namespace(),"object-container"));e.attr("transform","translate(0,0)"),n.attr("transform","translate(0,0)"),r.attr("transform","translate(0,0)")},f},O.violin=function(t){var e,n,a,i,l,u,d,g,p="horizontal",m=!0,b=!0,w=function(t,n){return e[t]},A=function(t,n){return d3.descending(e[t],e[n])},M=d3.scaleLinear(),E=.5,O=.05,L=50,C=100,F=2,Q=j(),B=function(t,e,n,o,a){var i=d3.scaleLinear().domain([o,a]).range([-.25,.05]),l=r(n.replace("#",""),i(t)),c="stroke"==e?0:.25;return r(l.replace("#",""),c)},q=3,V=2,D="transparent",_="d3sm-violin",K="violin",Y=1e3,P=d3.easeExp,X=["Q0","Q1","Q2","Q3","Q4"],R=z().keys([X[4],X[3],X[2],X[1],X[0]]),W=z(),N=function(t,e){return e.points},T=function(t,e){return e[t].value};function G(){var r,u,w,j="horizontal"==p,G=v(t,_,{x:0,y:0,width:n,height:a},D),H=void 0==i?d3.keys(e).sort(A):i;l=f(H);var Z=function(){var t,e,n=!0,r=["Q0","Q1","Q2","Q3","Q4"];function a(a,i){var l=i[a],c=t(a,l),u=d3.keys(c),s=u.map(function(t,n){return e(t,c)}),d=o(s,r),f=d3.histogram()(s),h=f.map(function(t){return t.length}),g=n?{x:0,y:d3.min(s)}:{x:d3.min(s),y:0},p=n?{x:0,y:d3.max(s)}:{x:d3.max(s),y:0},m=f.map(function(t,e){return n?{y:t.length?d3.median(t):d3.median([t.x0,t.x1]),x:h[e]}:{x:t.length?d3.median(t):d3.median([t.x0,t.x1]),y:h[e]}});m=[g].concat(m).concat([p]),l.binned=f,l.frequencies=h,l.contour=m,l.quartiles=d,l.pointKeys=u,l.pointValues=s}return a.horizontalQ=function(t){return arguments.length?(n=t,a):n},a.quartileKeys=function(t){return arguments.length?(r=t,a):r},a.violinPointsExtractor=function(e){return arguments.length?(t=e,a):t},a.violinPointValueExtractor=function(t){return arguments.length?(e=t,a):e},a}().horizontalQ(j).quartileKeys(X).violinPointsExtractor(N).violinPointValueExtractor(T);l.map(function(t,n){Z(t,e)});var I=l.length,J=(r=[]).concat.apply(r,S(l.map(function(t,n){return e[t].quartiles[X[0]]}))),U=(u=[]).concat.apply(u,S(l.map(function(t,n){return e[t].quartiles[X[X.length-1]]}))),$=[Math.min.apply(Math,S(J))-E,Math.max.apply(Math,S(U))+E];M.domain($).range(j?[0,a]:[0,n]);var tt=j?n:a;d=y(tt,I,L,C,O,m),g=x(H,tt,d,I,O,m),k().horizontalQ(j).scale(M).moveby("category").numberOfObjects(I).objectClass(K).objectSize(d).spacerSize(g).transitionDuration(Y).easeFunc(P).namespace(_)(G,H,0);var et=[];G.selectAll("g:not(.to-remove)."+K).each(function(t,e){s(l,t)&&et.push(Number(d3.select(this).attr("parent-index")))}),Q="index"==Q.colorBy()?Q.dataExtent([0,Math.max.apply(Math,et)]):Q.dataExtent($);var nt=Math.max.apply(Math,S((w=[]).concat.apply(w,S(l.map(function(t,n){return d3.max(e[t].frequencies)}))))),rt=d3.scaleLinear().domain([0,nt]).range([0,d/2]),ot=d3.line().x(function(t,e){return j?-rt(t.x):M(t.x)}).y(function(t,e){return j?M($[1])-M(t.y):-rt(t.y)}).curve(d3.curveBasis),at=d3.line().x(function(t,e){return j?rt(t.x):M(t.x)}).y(function(t,e){return j?M($[1])-M(t.y):rt(t.y)}).curve(d3.curveBasis);G.selectAll("g:not(.to-remove)."+K).each(function(t,n){var r=d3.select(this),o=e[t];if(s(l,t)){n=void 0==r.attr("parent-index")?n:r.attr("parent-index");var a=Q(t,o,n,"fill"),i=Q(t,o,n,"stroke"),u=c(r,"g","area"),f=c(u,"path","left"),g=c(u,"path","right"),p=c(r,"g","quarts"),m=(c(p,"line","q3"),c(p,"line","q1"),o.quartiles[X[3]],o.quartiles[X[2]]);if(o.quartiles[X[1]],r.attr("transform",j?"translate("+d/2+",0)":"translate(0,"+d/2+")"),f.transition().duration(Y).attr("d",function(t,e){return ot(o.contour)}).attr("fill",a).attr("stroke",i).attr("stroke-width",F),g.transition().duration(Y).attr("d",function(t,e){return at(o.contour)}).attr("fill",a).attr("stroke",i).attr("stroke-width",F),u.node().addEventListener("mouseover",function(t,e){G.selectAll("g."+K).style("opacity",.2),r.style("opacity",1),f.attr("stroke-width",2*F),g.attr("stroke-width",2*F)}),u.node().addEventListener("mouseout",function(t,e){G.selectAll("g."+K).style("opacity",1),f.attr("stroke-width",F),g.attr("stroke-width",F)}),b){var v=c(r,"g","points"),y=v.selectAll(".point").data(o.pointKeys);y.on("mouseover",null),y.exit().transition().ease(P).duration(Y).attr("r",0).attr("cy",j?M($[1])-M(m):rt(0)).attr("cx",j?rt(0):M(m)).remove();var x=y.enter().append("circle").attr("class","point").attr("r",0).attr("cx",j?0:M(m)).attr("cy",j?M(m):0);y=y.merge(x),z().selection(y).data(N(t,o)).header(W.header()).keys(W.keys()).values(W.values())();var k=d3.min(o.pointValues),w=d3.max(o.pointValues);y.transition().duration(Y).ease(P).attr("r",q).attr("cy",function(t,e){var n=o.pointValues[e];if(j)return M($[1])-M(n);var r=h(o.binned,n),a=Math.random(),i=rt(a*o.frequencies[r]*.5);return Math.random()>.5?i:-i}).attr("cx",function(t,e){var n=o.pointValues[e];if(j){var r=h(o.binned,n),a=Math.random(),i=rt(a*o.frequencies[r]*.5);return Math.random()>.5?i:-i}return M(n)}).attr("stroke",function(t,e){return t=o.pointValues[e],B(t,"stroke",i,k,w)}).attr("fill",function(t,e){return t=o.pointValues[e],B(t,"fill",i,k,w)}).attr("stroke-width",V),v.selectAll("circle.point").on("mouseover",function(t,e){G.selectAll("g."+K).style("opacity",.2),r.style("opacity",1),f.attr("stroke-width",2*F),g.attr("stroke-width",2*F),G.selectAll(".point").style("opacity",.2),d3.select(this).style("opacity",1).attr("r",2*q).attr("stroke-width",2*V)}),v.selectAll("circle.point").on("mouseout",function(t,e){var n=document.createEvent("SVGEvents");n.initEvent("mouseout",!0,!0),u.node().dispatchEvent(n),G.selectAll(".point").style("opacity",1),d3.select(this).attr("stroke-width",V).attr("r",q)})}else cV.selectAll(".point").transition().duration(Y).ease(P).attr("r",0).attr("cy",j?M($[1])-M(m):rt(0)).attr("cx",j?rt(0):M(m)).remove()}}),R.selection(G.selectAll("g:not(.to-remove)."+K+" .area")),void 0==R.data()&&R.data(e),R(),void 0==R.values()&&R.values([function(t,e){return t.quartiles[e]},function(t,e){return t.quartiles[e]},function(t,e){return t.quartiles[e]},function(t,e){return t.quartiles[e]},function(t,e){return t.quartiles[e]}])}return G.violinPointsExtractor=function(t){return arguments.length?(N=t,G):N},G.violinPointValueExtractor=function(t){return arguments.length?(T=t,G):T},G.selection=function(e){return arguments.length?(t=e,G):t},G.data=function(t){return arguments.length?(e=t,G):e},G.orient=function(t){return arguments.length?(p=t,G):p},G.spaceX=function(t){return arguments.length?(n=t,G):n},G.spaceY=function(t){return arguments.length?(a=t,G):a},G.overflowQ=function(t){return arguments.length?(m=t,G):m},G.pointsQ=function(t){return arguments.length?(b=t,G):b},G.grouping=function(t){return arguments.length?(i=t,G):i},G.valueExtractor=function(t){return arguments.length?(w=t,G):w},G.sortingFunction=function(t){return arguments.length?(A=t,G):A},G.scale=function(t){return arguments.length?(M=t,G):M},G.domainPadding=function(t){return arguments.length?(E=t,G):E},G.objectSpacer=function(t){return arguments.length?(O=t,G):O},G.minObjectSize=function(t){return arguments.length?(L=t,G):L},G.maxObjectSize=function(t){return arguments.length?(C=t,G):C},G.objectStrokeWidth=function(t){return arguments.length?(F=t,G):F},G.colorFunction=function(t){return arguments.length?(Q=t,G):Q},G.pointColorFunc=function(t){return arguments.length?(B=t,G):B},G.pointRadius=function(t){return arguments.length?(q=t,G):q},G.pointStrokeWidth=function(t){return arguments.length?(V=t,G):V},G.backgroundFill=function(t){return arguments.length?(D=t,G):D},G.namespace=function(t){return arguments.length?(_=t,G):_},G.objectClass=function(t){return arguments.length?(K=t,G):K},G.transitionDuration=function(t){return arguments.length?(Y=t,G):Y},G.easeFunc=function(t){return arguments.length?(P=t,G):P},G.quartileKey=function(t){return arguments.length?(quartileKey=t,G):quartileKey},G.quartileKeys=function(t){return arguments.length?(X=t,G):X},G.violinKeys=function(t){return arguments.length?(l=t,G):l},G.violinValues=function(t){return arguments.length?(u=t,G):u},G.objectSize=function(t){return arguments.length?(d=t,G):d},G.spacerSize=function(t){return arguments.length?(g=t,G):g},G.tooltip=function(t){return arguments.length?(R=t,G):R},G.pointsTooltip=function(t){return arguments.length?(W=t,G):W},G},O.numericLegend=function(t){var e,n,r=0,o=1,l=j(),u="d3sm-linear-vertical-gradient",s=12,d="transparent",f="black",h=2;function g(){var g=v(t,u,{x:0,y:0,width:e,height:n},d),p=c(c(t,"defs"),"linearGradient").attr("x1","0%").attr("y1","100%").attr("x2","0%").attr("y2","0%").attr("id",a(u,"numerical-legend-gradient"));l.dataExtent([r,o]).colorBy("value").valueExtractor(function(t,e,n){return e}),p.selectAll("stop").data(l.colors()).enter().append("stop").attr("offset",function(t,e){return e/(l.colors().length-1)}).attr("stop-color",function(t){return t});var m=c(g,"rect","legend").attr("transform","translate(0,"+s+")").style("fill","url(#"+a(u,"numerical-legend-gradient")+")").attr("x",0).attr("y",0).attr("width",e).attr("height",n-2*s).on("mousemove",function(t,e){!function(t,e,n,d){var g=d3.scaleLinear().domain([0,n.attr("height")]).range([o,r]),p=d3.mouse(n.node()),m=i(g(p[1]),h),v=l(void 0,m,void 0,"stroke"),y=l(void 0,m,void 0,"fill");c(c(d3.select("body"),"div",a(u,"legend-tooltip")).attr("id",a(u,"legend-tooltip")).style("position","absolute").style("left",d3.event.pageX+15+"px").style("top",d3.event.pageY+15+"px").style("background-color",y).style("border-color",v).style("min-width",s*(String(o).split(".")[0].length+3)+"px").style("min-height",s*(String(o).split(".")[0].length+3)+"px").style("border-radius","50%").style("border-radius","5000px").style("display","flex").style("justify-content","center").style("text-align","middle").style("padding","2px").style("border-style","solid").style("border-width",2),"div").text(m).style("color",f).style("align-self","center")}(0,0,m,d3.select(this))}).on("mouseout",function(t,e){d3.select("#"+a(u,"legend-tooltip")).remove()});c(g,"text","min").text(i(r,2)).attr("text-anchor","middle").attr("font-size",s+"px").attr("transform",function(t,r){return"translate("+e/2+","+(n-s/4)+")"}),c(g,"text","max").text(i(o,2)).attr("text-anchor","middle").attr("font-size",s+"px").attr("transform",function(t,n){return"translate("+e/2+","+s+")"})}return g.min=function(t){return arguments.length?(r=t,g):r},g.max=function(t){return arguments.length?(o=t,g):o},g.spaceX=function(t){return arguments.length?(e=t,g):e},g.spaceY=function(t){return arguments.length?(n=t,g):n},g.namespace=function(t){return arguments.length?(u=t,g):u},g.fontSize=function(t){return arguments.length?(s=t,g):s},g.backgroundFill=function(t){return arguments.length?(d=t,g):d},g.colorFunction=function(t){return arguments.length?(l=t,g):l},g.textColor=function(t){return arguments.length?(f=t,g):f},g.roundTo=function(t){return arguments.length?(h=t,g):h},g},O.categoricLegend=function(t){var e,n,r,o,a,i="horizontal",l=!1,u=function(t,e){return n[t]},s=function(t,e){return d3.ascending(t,e)},d=.05,h=10,g=100,p=2,m=j(),b="transparent",w="d3sm-legend",S="legend",z=1e3,A=d3.easeExp;function M(){var n="horizontal"==i,u=!n,j=v(t,w,{x:0,y:0,width:r,height:o},b);m.dataExtent([0,e.length-1]).colorBy("categories").categoryExtractor(function(t,e,n){return e});var M=Math.min(r,o),E=e.length,O=void 0==a?e.sort(s):a,L=f(O),C=n?r:o,F=y(C,E,h,g,d,l),Q=x(L,C,F,E,d,l);k().horizontalQ(n).scale(scale).moveby("category").numberOfObjects(E).objectClass(S).objectSize(F).spacerSize(Q).transitionDuration(z).easeFunc(A).namespace(w)(j,O,0),M=Math.min(F,r,o)/2-p,j.selectAll("g:not(.to-remove)."+S).each(function(t,e){var o=d3.select(this),a=c(o,"circle"),i=m(void 0,t,e,"fill"),l=m(void 0,t,e,"stroke"),s=n?M+p:(r-2*M)/2+M,d=u?M+p:(r-2*M)/2+M;a.attr("r",M).attr("cx",s).attr("cy",d).attr("fill",i).attr("stroke",l).attr("stroke-width",p);var f=c(o,"text");f.text(t).attr("text-anchor","middle").attr("transform",function(t,e){return"translate("+s+","+(d+f.node().getBoundingClientRect().height/4)+")"})})}return M.categories=function(t){return arguments.length?(e=t,M):e},M.selection=function(e){return arguments.length?(t=e,M):t},M.data=function(t){return arguments.length?(n=t,M):n},M.orient=function(t){return arguments.length?(i=t,M):i},M.spaceX=function(t){return arguments.length?(r=t,M):r},M.spaceY=function(t){return arguments.length?(o=t,M):o},M.overflowQ=function(t){return arguments.length?(l=t,M):l},M.grouping=function(t){return arguments.length?(a=t,M):a},M.valueExtractor=function(t){return arguments.length?(u=t,M):u},M.sortingFunction=function(t){return arguments.length?(s=t,M):s},M.objectSpacer=function(t){return arguments.length?(d=t,M):d},M.minObjectSize=function(t){return arguments.length?(h=t,M):h},M.maxObjectSize=function(t){return arguments.length?(g=t,M):g},M.bubbleStrokeWidth=function(t){return arguments.length?(p=t,M):p},M.colorFunction=function(t){return arguments.length?(m=t,M):m},M.backgroundFill=function(t){return arguments.length?(b=t,M):b},M.namespace=function(t){return arguments.length?(w=t,M):w},M.objectClass=function(t){return arguments.length?(S=t,M):S},M.transitionDuration=function(t){return arguments.length?(z=t,M):z},M.easeFunc=function(t){return arguments.length?(A=t,M):A},M},O.lasso=E,O.lassoWidget=function(t){var e,n,r,o,i,l,u,s,d="d3sm-lasso",f=(t=t,d3.line().x(function(t,e){return t[0]}).y(function(t,e){return t[1]}).curve(d3.curveLinearClosed),j());function h(t,e){console.log(t,e)}function g(){d3.select("html").select("style."+d+"lasso-widget").empty()&&d3.select("html").append("style").classed(d+"lasso-widget",!0).html("."+a(d,"data-table")+"{ height:100px; overflow:auto; }")}function p(t){console.log(t)}function m(){p((t.select("."+a(d,"tab-list")),t.select("."+a(d,"tab-content")).selectAll("."+a(d,"data-table")),{}))}function v(e,n){var r,o;t.select("#"+a(d,"tab",e)),t.select("#"+a(d,"tab","pane",e)),function(){var e=t.select("."+a(d,"tab-list")),n=(t.select("."+a(d,"tab-content")),e.selectAll("."+a(d,"tab")));if(0==n.size());else{var r=n.nodes()[n.size()-1];d3.select(r).select("a").attr("href")}}(),r=t.select("."+a(d,"tab-list")),o=t.select("."+a(d,"tab-content")),r=r.selectAll("."+a(d,"tab")),o=o.selectAll("."+a(d,"tab","pane")),r.each(function(t,e){d3.select(this).datum(e).attr("id",a(d,"tab",e)).select("a").attr("href","#"+a(d,"tab","pane",e)).text(function(t,n){var r=d3.select(this).text();return"Group"==r.split(" ")[0]?"Group "+e:r})}),o.each(function(t,e){d3.select(this).datum(e).attr("id",a(d,"tab","pane",e)),c(d3.select(this),"p","lead").text(function(t,n){var r=d3.select(this).text();return"Group"==r.split(" ")[0]?"Group "+e:r}),c(d3.select(this),"button","remove-btn").on("click",v)})}function y(t,i){var h=c(t,"p","lead").text("Group "+i),g=c(c(t,"div","table-responsive").attr("class",a(d,"data-table")),"table","table").classed("table-sm",!0).classed("table-hover",!0),p=(c(g,"caption","caption").html("List of selected"),c(t,"div","text-right")),m=i%f.colors().length;i%2!=0&&(m=f.colors().length-1-m);var y,k,w,S=function(t){var e=c(t,"button","lasso-btn").classed("btn btn-info",!0).classed(d,!0);return c(e,"i","fa").classed("fa-hand-pointer-o",!0),c(e,"span").text("Lasso select"),e}(p),j=(y=i,k=f.colors()[m],E().namespace(d).svg(e).objectClass(o).chartContainer(n).objectContainer(r).instance(y).color(k).yScale(s).xScale(u));!function(t,e,r,o){r.eventCatcher(t),t.node().addEventListener("click",x),t.datum([r]).on(a(d,"render"),function(){d3.select(this).datum()[0].draw()}).on(a(d,"drag"),function(t){var e=n.selectAll(".in-lasso-"+t[0].instance()).nodes(),a=e.map(function(t,e){var n=l(d3.select(t).datum());return n.__node=t,n});b(o,a,r)}),e.on("click",function(){r.allPoints([]),r.currentPoints([]),t.dispatch(a(d,"drag"))})}(S,function(t){var e=c(t,"button","clear-btn").classed("btn btn-light",!0).classed(d,!0);return c(e,"i","fa").classed("fa-eraser",!0),c(e,"span").text("Clear selection"),e}(p),j,g),c(w=c(p,"button","remove-btn").classed("btn btn-danger",!0).on("click",v),"i","fa").classed("fa-trash-o",!0),c(w,"span").text("Remove group"),h.on("mouseover",function(){j.draw()}),h.on("mouseout",function(){j.remove()})}function x(){var e=d3.select(this),n=e.datum()[0];n.toggle();var r=n.activeQ();function o(t){t.target!=e.node()&&t.target!=e.select("span").node()&&t.target!=e.select("i").node()&&(n.toggle(!1),e.classed("btn-info",!0),e.classed("btn-warning",!1),e.select("span").text("Lasso select"))}n.activeQ()?(e.classed("btn-info",!r),e.classed("btn-warning",r),e.select("span").text("Lasso select (active)"),t.selectAll("."+d+".lasso-btn").dispatch(a(d,"render")),d3.select("html").node().addEventListener("mousedown",o)):(e.classed("btn-info",!r),e.classed("btn-warning",r),e.select("span").text("Lasso select"),d3.select("html").node().removeEventListener("mousedown",o))}function b(t,e,n){var r=c(c(t,"thead"),"tr"),o=c(t,"tbody"),a=function(t,e){var n=d3.keys(e[0]).filter(function(t){return"__node"!=t});n.length>0&&n.push("remove");var r=t.selectAll("th");return(r=r.data(n)).exit().remove(),r=r.merge(r.enter().append("th").attr("scope","col")).text(function(t,e){return t}),n}(r,e);(function(t,e){var n=t.selectAll("tr");return(n=n.data(e)).exit().remove(),n=n.merge(n.enter().append("tr"))})(o,e).each(function(r,o){var i=d3.select(this);!function(t,e,n,r,o,a){(t=t.data(e)).exit().remove(),(t=t.merge(t.enter().append("td"))).html(function(t,e){return"remove"!=t?n[t]:""}).classed("text-left",function(t,e){return"remove"==t}),t.select("i.fa-close").on("click",function(t,e){var i=n.__node,l=d3.select(i);l.classed("in-lasso",!1),l.classed("in-lasso-"+o.instance(),!1),r.map(function(t,e){t.__node==i&&(r.splice(e,1),b(a,r,o))})})}(i.selectAll("td"),a,r,e,n,t),i.on("mouseover",function(t,e){n.applyObjectAttributes(d3.select(t.__node),!0)}).on("mouseout",function(t,e){n.applyObjectAttributes(d3.select(t.__node),!1)})})}function k(e,n){var r=t.select("."+a(d,"tab-list")),o=t.select("."+a(d,"tab-content")),l=function(){for(var e=t.select("."+a(d,"tab-list")).selectAll("li").size()-3,n="Group "+e,r=[];r.includes(n);)n="Group "+(e+=1);return e}();if(r.selectAll("."+a(d,"tab")).size()!=i){!function(t,e){var n=t.insert("li",".right-aligned-tabs").classed("nav-item",!0).classed("alert",!0).classed("alert-secondary",!0).classed("alert-dismissable",!0).classed("fade",!0).classed("show",!0).attr("role","alert").attr("id",a(d,"tab",e)).classed(a(d,"tab"),!0);c(n,"a").attr("data-toggle","tab").text("Group "+e).attr("href","#"+a(d,"tab","pane",e)).on("dblclick",function(t,e){var n=d3.select(this);n.attr("contenteditable",!0),d3.select(n.node().parentNode).classed("alert-secondary",!1).classed("alert-warning",!0)}).on("blur",function(t,e){var n=d3.select(this);n.attr("contenteditable",!1),d3.select(n.node().parentNode).classed("alert-secondary",!0).classed("alert-warning",!1)}).on("input",function(t,e){var n=d3.select(this),r=n.text();d3.select(n.attr("href")).select("p.lead").text(r)})}(r,l);var u=function(t,e){return t.append("div").datum(e).attr("role","tabpanel").classed("tab-pane",!0).classed("text-left",!0).classed(a(d,"tab","pane"),!0).attr("id",a(d,"tab","pane",e))}(o,l);y(u,l),u.attr("id"),t.select("."+a(d,"tab-content"))}else h("only "+i+" allowed.","warning")}return w=c(t,"div","card"),S=c(c(w,"div","card-header"),"ul","nav").classed("nav-tabs card-header-tabs",!0).classed(a(d,"tab-list"),!0).attr("role","tablist"),z=c(c(w,"div","card-body"),"div","tab-content").classed(a(d,"tab-content"),!0),A=c(S,"div","right-aligned-tabs").classed("nav ml-auto ",!0),function(t){c(c(c(t,"li",a(d,"plus-tab")).classed("ml-auto",!0).classed("nav-item",!0).on("click",k),"a","nav-link"),"i","fa").classed("fa-plus fa-2x text-success",!0)}(A),function(t){c(c(c(t,"li",a(d,"send-tab")).classed("ml-auto",!0).classed("nav-item",!0).on("click",m),"a","nav-link"),"i","fa").classed("fa-paper-plane-o fa-2x text-primary",!0)}(A),function(t){c(c(c(t,"li",a(d,"close-tab")).classed("ml-auto",!0).classed("nav-item",!0).on("click",function(){d3.mouse(d3.select("html").node())}),"a","nav-link"),"i","fa").classed("fa-window-close-o fa-2x text-danger",!0)}(A),c(c(z,"div","tab-pane").classed(a(d,"default-tab"),!0).classed("active",!0).classed("text-left",!0),"div").html("Click to add a new group.
Click to submit for re-analysis.
Click to close the dataselect widget."),g.objectClass=function(t){return arguments.length?(o="."+t.replace(".",""),g):o},g.svg=function(t){return arguments.length?(e=t,g):e},g.submit=function(t){return arguments.length?(p=t,g):p},g.maxNumberOfGroups=function(t){return arguments.length?(i=t,g):i},g.onError=function(t){return arguments.length?(h=t,g):h},g.objectContainer=function(t){return arguments.length?(r=t,g):r},g.chartContainer=function(t){return arguments.length?(n=t,g):n},g.dataExtractor=function(t){return arguments.length?(l=t,g):l},g.xScale=function(t){return arguments.length?(u=t,g):u},g.yScale=function(t){return arguments.length?(s=t,g):s},g;var w,S,z,A},O.selectFilter=A,O.upset=function(t){var e,n,r,o,i,l,u,s,f,h,g,p="horizontal",m=!1,b=20,w=50,S=2,j="transparent",z="d3sm-upset",A="upset",M=1e3,E=d3.easeExp,O=function(t,n){return e[t].set},L=function(t,n){return e[t].intersection},C=function(t,n){return e[t].elements},F=.05,Q=.05,B=function(t,e){return i.indexOf(O(t))-i.indexOf(O(e))},q=function(t,e){return l.indexOf(L(t))-l.indexOf(L(e))};function V(){var S="horizontal"==p,C=!S,V=v(t,z,{x:0,y:0,width:n,height:r},j);o=d3.keys(e),i=d(o.map(O)).sort(),l=d(o.map(L)).sort().sort(function(t,e){return t.split(";").length-e.split(";").length}),S?o.sort(function(t,e){return q(t,e)||B(t,e)}):o.sort(function(t,e){return B(t,e)||q(t,e)});var D=S?l:i,_=S?i:l,K=S?D.length:_.length,Y=S?_.length:D.length;h=y(n,K,b,w,F,m),s=y(r,Y,b,w,Q,m),g=x(D,n,h,K,F,m),f=x(_,r,s,Y,Q,m);var P=k().horizontalQ(!1).moveby("category").numberOfObjects(Y).objectSize(s).spacerSize(f).transitionDuration(M).easeFunc(E),X=k().horizontalQ(!0).moveby("category").numberOfObjects(K).objectClass(A).objectSize(h).spacerSize(g).transitionDuration(M).easeFunc(E);C?(X.objectClass(A),P.namespace("across").objectClass(a(A,"across")),P(V,_,0),V.selectAll("g."+a(A,"across")).each(function(t,e){X(d3.select(this),D,0)})):(X.namespace("across").objectClass(a(A,"across")),P.objectClass(A),X(V,D,0),V.selectAll("g."+a(A,"across")).each(function(t,e){P(d3.select(this),_,0)}));var R=V.selectAll("g:not(.to-remove)."+A),W={};o.map(function(t,e){W[O(t)+"::"+L(t)]=t}),R.data(o),R.each(function(t,n){var r=d3.select(this);if(void 0!=t){e[t];var o=O(t,n),i=L(t,n);r.classed(i,!0),r.classed(o,!0),c(r,"circle",a(A,"circle")).attr("cx",h/2).attr("cy",s/2).attr("r",void 0==u?Math.min(h,s)/2:u).attr("fill",i.includes(o)?"black":"rgb(233,233,233)").attr("stroke","black").attr("in-intersection",i.includes(o))}})}return V.selection=function(e){return arguments.length?(t=e,V):t},V.data=function(t){return arguments.length?(e=t,V):e},V.orient=function(t){return arguments.length?(p=t,V):p},V.spaceX=function(t){return arguments.length?(n=t,V):n},V.spaceY=function(t){return arguments.length?(r=t,V):r},V.overflowQ=function(t){return arguments.length?(m=t,V):m},V.minObjectSize=function(t){return arguments.length?(b=t,V):b},V.maxObjectSize=function(t){return arguments.length?(w=t,V):w},V.circleStrokeWidth=function(t){return arguments.length?(S=t,V):S},V.backgroundFill=function(t){return arguments.length?(j=t,V):j},V.namespace=function(t){return arguments.length?(z=t,V):z},V.objectClass=function(t){return arguments.length?(A=t,V):A},V.transitionDuration=function(t){return arguments.length?(M=t,V):M},V.easeFunc=function(t){return arguments.length?(E=t,V):E},V.cellKeys=function(t){return arguments.length?(o=t,V):o},V.setValues=function(t){return arguments.length?(i=t,V):i},V.intersectionValues=function(t){return arguments.length?(l=t,V):l},V.xObjectSpacer=function(t){return arguments.length?(F=t,V):F},V.yObjectSpacer=function(t){return arguments.length?(Q=t,V):Q},V.radius=function(t){return arguments.length?(u=t,V):u},V.setExtractor=function(t){return arguments.length?(O=t,V):O},V.intersectionExtractor=function(t){return arguments.length?(L=t,V):L},V.elementExtractor=function(t){return arguments.length?(C=t,V):C},V.setKeySortingFunction=function(t){return arguments.length?(B=t,V):B},V.intersectionKeySortingFunction=function(t){return arguments.length?(q=t,V):q},V.yObjectSize=function(t){return arguments.length?(s=t,V):s},V.ySpacerSize=function(t){return arguments.length?(f=t,V):f},V.xObjectSize=function(t){return arguments.length?(h=t,V):h},V.xSpacerSize=function(t){return arguments.length?(g=t,V):g},V.intersectionTotals=function(){var t={};return l.map(function(e,n){t[e]={total:0}}),o.map(function(e,n){var r=C(e,n);0==t[L(e,n)].total&&(Array.isArray(r)?(t[L(e,n)].total+=r.length,t[L(e,n)].values=r):t[L(e,n)].total+=r)}),t},V.setTotals=function(){var t={};return i.map(function(e,n){t[e]={total:0}}),o.map(function(e,n){var r=C(e,n);Array.isArray(r)?t[O(e,n)].total+=r.length:t[O(e,n)].total+=r}),t},V},O.filterTable=function(t){var e,n="d3sm-filter-table",r=function(t,e,n){return n};function o(){var o=c(t,"div","filter-table").classed(a(n,"container"),!0),u=c(o,"div","filter-input-group").classed("input-group",!0),s=(c(c(c(u,"div","input-group-prepend"),"span","input-group-text").classed("search-button",!0),"i","fa fa-search"),c(u,"input","form-control").attr("placeholder","filter").attr("type","text")),d=c(c(u,"div","input-group-append"),"a","close-button").classed("btn btn-outline-secondary",!0),f=(c(d,"i","fa fa-close"),d),h=c(c(o,"div","table-responsive"),"table","table").classed("table-hover",!0).classed("table-bordered",!0).classed("table-striped",!0),g=c(c(h,"thead").classed("thead-dark",!0),"tr"),p=c(h,"tbody"),m=d3.keys(e),v=d3.keys(e[m[0]]);(function(t,e){var n=t.selectAll("th");(n=n.data(e)).exit().remove(),(n=n.merge(n.enter().append("th"))).attr("scope","col").text(function(t,e){return t})})(g,v),l(i(p,m),v),s.on("input",function(t,n){var o,a=s.property("value"),c=new RegExp(a,"gi");""==a?o=m:(o=[],m.map(function(t,n){var a=e[t],i=v.map(function(t,e){return r(a,t,a[t])}).join("").match(c);null==i||""==i.join("")||o.push(t)})),l(i(p,o),v)}),f.on("click",function(t,e){s.property("value","").dispatch("input")})}function i(t,e){var n=t.selectAll("tr");return(n=n.data(e)).exit().remove(),n=n.merge(n.enter().append("tr"))}function l(t,n){return t.each(function(t,o){var a=e[t],i=d3.select(this).selectAll("td");(i=i.data(n)).exit().remove(),(i=i.merge(i.enter().append("td"))).attr("scope",function(t,e){return 0==e}).html(function(t,e){return r(a,t,a[t])})}),t}return o.data=function(t){return arguments.length?(e=t,o):e},o.namespace=function(t){return arguments.length?(n=t,o):n},o.fieldFunction=function(t){return arguments.length?(r=t,o):r},o},O.uniqueElements=e,O.getTranslation=n,O.modifyHexidecimalColorLuminance=r,O.tickRange=u,O.quartiles=o,O.extractViolinValues=function(t,e,n,r,a,i){var l={};return t.map(function(t,c){var u=n(t,c,e),s=d3.histogram()(u),d=s.map(function(t){return t.length}),f=r?{y:d3.min(u),x:0}:{x:d3.min(u),y:0},h=r?{y:d3.max(u),x:0}:{x:d3.max(u),y:0},g=s.map(function(t,e){return r?{y:t.length?d3.median(t):d3.median([t.x0,t.x1]),x:d[e]}:{x:t.length?d3.median(t):d3.median([t.x0,t.x1]),y:d[e]}}),p=o(u,i),m={values:u,binned:s,frequencies:d,points:[f].concat(g).concat([h])};m[a]=p,l[t]=m}),l},O.hypenate=a,O.round=i,O.getContainingSVG=l,O.interpolateColors=function(){return d3.interpolateRgbBasis(arguments)},O.truncateText=function(t,e,n,r,o,a){var i=t.node().getBoundingClientRect();for(t.text(e);Math.max(i.width,i.height)>o-r&&(e=(e=String(e)).slice(0,e.length-1),t.text(e+"..."),i=t.node().getBoundingClientRect(),0!=e.length););},O.safeSelect=c,O.whichBin=h,O.unique=d,O.flatten=f,O.setupStandardChartContainers=function(t,e,n){var r=arguments.length>3&&void 0!==arguments[3]?arguments[3]:{top:.01,bottom:.01,left:.01,right:.01},o=arguments.length>4&&void 0!==arguments[4]?arguments[4]:{w:.8,h:.6},a=arguments.length>5&&void 0!==arguments[5]?arguments[5]:{y:.1,x:.1},i=arguments.length>6&&void 0!==arguments[6]?arguments[6]:{x:0,margin:0,pos:"left"};void 0==n&&(n={w:window.innerWidth,h:window.Height});var l={w:n.w*o.w,h:n.h*o.h},c={top:r.top*l.h,bottom:r.bottom*l.h,left:r.left*l.w,right:r.right*l.w},u=l.w-c.left-c.right,s=l.h-c.top-c.bottom,d={x:s*a.x,y:u*a.y},f={x:u-d.y-i.x-2*i.margin,y:s-d.x},h={x:i.margin+c.left+("left"==i.pos?0:f.x+d.y),y:c.top,w:i.x,h:f.y},g={x:d.y+c.left+("left"==i.pos?i.x+2*i.margin:0),y:c.top,w:d.y,h:f.y},p={x:d.y+c.left+("left"==i.pos?i.x+2*i.margin:0),y:c.top,w:f.x,h:f.y},m={x:d.y+c.left+("left"==i.pos?i.x+2*i.margin:0),y:c.top+f.y,w:f.x,h:d.x};return n=d3sm.safeSelect(t,"svg",e).style("width",l.w+"px").style("height",l.h+"px"),a=d3sm.safeSelect(n,"g",d3sm.hypenate(e,"axes")),i=d3sm.safeSelect(n,"g",d3sm.hypenate(e,"legend")).attr("transform","translate("+h.x+","+h.y+")"),{svg:{selection:n,rect:l},plot:{selection:d3sm.safeSelect(n,"g",d3sm.hypenate(e,"plot")).attr("transform","translate("+p.x+","+p.y+")"),rect:p},xAxis:{selection:d3sm.safeSelect(a,"g",d3sm.hypenate(e,"x-axis")).attr("transform","translate("+m.x+","+m.y+")"),rect:m},yAxis:{selection:d3sm.safeSelect(a,"g",d3sm.hypenate(e,"y-axis")).attr("transform","translate("+g.x+","+g.y+")"),rect:g},legend:{selection:i,rect:h}}},O.log=m,O.warn=function(t,e,n){!0===window.d3sm.debugQ&&console.warn("%c[d3sm::"+t+"]:\t"+e,["background: #ffd53e","border-radius: 5000px","padding: 0px 2px","font-size: 14px"].join(";")),console.table(n)},O.info=function(t,e,n){window.d3sm.debugQ&&console.info("%c[d3sm::"+t+"]:\t"+e,["background: #009ccd","border-radius: 5000px","padding: 0px 2px","font-size: 14px"].join(";")),console.table(n)},O.error=function(t,e,n){window.d3sm.debugQ&&console.error("[d3sm::"+t+"]:\t"+e+"\t%o",n)},O.consoleGroup=g,O.consoleGroupEnd=p,O.resizeDebounce=function(t,e){var n=function(t,e,n){var r;return function(){var o=this,a=arguments,i=n&&!r;clearTimeout(r),r=setTimeout(function(){r=null,n||t.apply(o,a)},e),i&&t.apply(o,a)}}(function(){t()},e);window.addEventListener("resize",n)},O.debugQ=!1,window.d3sm=O,t.default=O,t}({}); -//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"d3sm.min.v0.0.4.js","sources":["../../src/scripts/modules/helpers.js","../../src/scripts/modules/array-functions.js","../../src/scripts/modules/utils.js","../../src/scripts/modules/grouping-spacer.js","../../src/scripts/modules/color-function.js","../../src/scripts/modules/tooltip.js","../../src/scripts/modules/select-filter.js","../../src/scripts/modules/lasso.js","../../src/scripts/modules/d3-prototypes.js","../../src/scripts/main.js","../../src/scripts/modules/axis.js","../../src/scripts/modules/bar.js","../../src/scripts/modules/bubble-heatmap.js","../../src/scripts/modules/heatmap.js","../../src/scripts/modules/box-whisker.js","../../src/scripts/modules/data-toggle.js","../../src/scripts/modules/scatter.js","../../src/scripts/modules/plot-zoom.js","../../src/scripts/modules/multi-plot-zoom.js","../../src/scripts/modules/violin.js","../../src/scripts/modules/numeric-legend.js","../../src/scripts/modules/categorical-legend.js","../../src/scripts/modules/lasso-widget.js","../../src/scripts/modules/upset.js","../../src/scripts/modules/filter-table.js"],"sourcesContent":["// import {hasQ} from './array-functions';\n/*******************************************************************************\n**                                                                            **\n**                                                                            **\n**                                HELPERS                                     **\n**                                                                            **\n**                                                                            **\n*******************************************************************************/\n/**\n* Helper function for Array.filter to get unique elements of the array\n* @param {*} value current value as mapping over array (self)\n* @param {number} index current index in the array\n* @param {Array} self passed array from Array.filter method\n* @returns {boolean} whether or not value is the first of its kind (i.e. indexOf(value) == index)\n*/\nexport function uniqueElements(value, index, self) { return self.indexOf(value) === index; }\n\n/**\n* Extracts x and y of translate from transform property\n* @param {string} transform transform property of svg element\n* @returns {number[]} x, y of translate(x, y)\n*/\nexport function getTranslation(transform) {\n  // Create a dummy g for calculation purposes only. This will never\n  // be appended to the DOM and will be discarded once this function\n  // returns.\n  var g = document.createElementNS('http://www.w3.org/2000/svg', 'g');\n  // Set the transform attribute to the provided string value.\n  transform = transform == undefined ? 'translate(0,0)' : transform;\n  g.setAttributeNS(null, 'transform', transform);\n  // consolidate the SVGTransformList containing all transformations\n  // to a single SVGTransform of type SVG_TRANSFORM_MATRIX and get\n  // its SVGMatrix.\n  var matrix = g.transform.baseVal.consolidate().matrix;\n  // As per definition values e and f are the ones for the translation.\n  return [matrix.e, matrix.f];\n}\n\n\n/**\n* Modifies luminance of hexidecimal number\n* @param {string} hex should be hexidecimal value with or without the proceeding octotrope\n* @param {number} lum value to increase or decrease luminosity by\n* @returns {string} updated hexidecimal value without the proceeding octotrope\n*/\nexport function modifyHexidecimalColorLuminance(hex, lum) {\n  // validate hex string\n  var hex = String(hex).replace(/[^0-9a-f]/gi, '');\n\n  if (hex.length < 6) {\n    hex = hex[0]+hex[0]+hex[1]+hex[1]+hex[2]+hex[2];\n\t}\n\tlum = lum || 0;\n\n\t// convert to decimal and change luminosity\n\tvar rgb = '#', c, i;\n\tfor (i = 0; i < 3; i++) {\n\t\tc = parseInt(hex.substr(i*2,2), 16);\n\t\tc = Math.round(Math.min(Math.max(0, c + (c * lum)), 255)).toString(16);\n\t\trgb += ('00'+c).substr(c.length);\n\t}\n\n\treturn rgb;\n}\n\n\n/**\n* @deprecated @see{@link tickRange}\n* @param {number} min\n* @param {number} max\n* @param {number} parts\n* @returns {number[]} array of length parts evenly partitioned between min and max\n*/\nexport function partitionRangeInto(min, max, parts) {\n  var diff = max - min\n  return Array(parts).map(function (e, i) { return min + diff / parts * i })\n}\n\n\n/**\n* Calculated the quartiles of the passed data and stores them with qKeys\n* @param {number[]} data list of numerical values\n* @param {string[]} [qKeys=['q0', 'q1', 'q2', 'q3', 'q4']] how returned object with quartiles should be stored\n* @returns {Object} with keys qKeys giving only the numerical values for the quartiles\n*/\nexport function quartiles(data, qKeys) {\n  var\n  q2 = d3.median(data),\n  lower = data.filter(x => x < q2),\n  upper = data.filter(x => x > q2),\n\n  q1 = d3.median(lower),\n  q1 = q1 == undefined ? q2 : q1,\n\n  q0 = d3.min(lower),\n  q0 = q0 == undefined ? q1 : q0,\n\n  q3 = d3.median(upper),\n  q3 = q3 == undefined ? q2 : q3,\n\n  q4 = d3.max(upper),\n  q4 = q4 == undefined ? q3 : q4,\n\n  k0 = 'q0', k1 = 'q1', k2 = 'q2', k3 = 'q3', k4 = 'q4',\n  obj = {}\n  if (qKeys!=undefined && qKeys.length == 5) { k0 = qKeys[0]; k1 = qKeys[1]; k2 = qKeys[2]; k3 = qKeys[3]; k4 = qKeys[4]; }\n  obj[k0] = q0; obj[k1] = q1; obj[k2] = q2; obj[k3] = q3; obj[k4] = q4;\n\n  return obj\n}\n\n\n/**\n* Helper function to get all values needed in making violin plots\n* @param {string[]} violinKeys\n* @param {number[]} data\n* @param {Function} valueExtractorFunction how to get values from data[violinKeys[i]]\n* @param {boolean} horizontalQ whether or not violins will be rendered horizontally or vertically\n* @param {string} qKey how the object containing the quartiles should be labeled as\n* @param {string[]} qKeys how each quartile should be labeled as\n* @returns {Object} required for @see{@link violin} containing keys values, binnned, frequencies, points, and quartiles\n* @see{@link quartiles}\n*/\nexport function extractViolinValues(\n  violinKeys,\n  data,\n  valueExtractorFunction,\n  horizontalQ,\n  qKey,\n  qKeys\n){\n  var obj = {}\n  violinKeys.map(function(k, i){\n     var d = valueExtractorFunction(k, i, data),\n     binned = d3.histogram()(d),\n     frequencies = binned.map(x=>x.length),\n     minPoint = horizontalQ ? {y: d3.min(d), x: 0} : {x: d3.min(d), y: 0},\n     maxPoint = horizontalQ ? {y: d3.max(d), x: 0} : {x: d3.max(d), y: 0},\n     points = binned.map(function(bin, i) {\n       return horizontalQ\n       ? {y: (bin.length) ? d3.median(bin): d3.median([bin.x0, bin.x1]), x: frequencies[i]}\n       : {x: (bin.length) ? d3.median(bin): d3.median([bin.x0, bin.x1]), y: frequencies[i]}\n     }),\n     quarts = quartiles(d, qKeys),\n     o = {\n       values: d,\n       binned: binned,\n       frequencies: frequencies,\n       points: [minPoint].concat(points).concat([maxPoint])\n     }\n     o[qKey] = quarts;\n     obj[k] = o;\n   });\n   return obj;\n}\n\n/**\n* Hypenates all strings together\n* @param {string[]} arguments\n* @returns {string} \"arg1-arg2-...-argn\"\n*/\nexport function hypenate(){ return Array.prototype.slice.call(arguments).join('-') }\n\n\n/**\n* Rounds decimals of number to precision\n* @param {number} number\n* @param {number} precision\n* @returns {number} rounded to precision\n*/\nexport function round(number, precision) {\n  var shift = function (number, precision, reverseShift) {\n    if (reverseShift) {\n      precision = -precision;\n    }\n    var numArray = ('' + number).split('e');\n    return +(numArray[0] + 'e' + (numArray[1] ? (+numArray[1] + precision) : precision));\n  };\n  return shift(Math.round(shift(number, precision, false)), precision, true);\n}\n\n/**\n* recursively ascends element.parentElement to find a svg tag\n* @param {Element} element\n* @returns {Element | undefined}\n*/\nexport function getContainingSVG(element) {\n  var parent = element.parentElement\n  var tag = parent.tagName.toLowerCase()\n  if (tag === 'svg') { return parent; }\n  if (tag === 'html') { return undefined; }\n  return getContainingSVG(parent);\n}\n\n/**\n* Maps arguments in to d3.interpolateRgbBasis\n* @param arguments\n* @returns {Function}\n*/\nexport function interpolateColors(){return d3.interpolateRgbBasis(arguments)}\n\n\n/**\n* Trys to reduce text to fit in specified area, made for tick labels as called by\n* @see{@link axis}\n* @param {d3.selection} t container for specific axis tick\n* @param {string} text to be the label of the passed axis tick\n* @param {boolean} orient of the axis, true is horizontal, false is vertical\n* @param {number} tickLength is the length of the text\n* @param {number} space is the amount of availble space for the text and the tick to fit in\n* @param {boolean} overflowQ whether or not allowed to go over the alloted space\n* @returns {none}\n*/\nexport function truncateText(t, text, orient, tickLength, space, overflowQ) {\n  var rect = t.node().getBoundingClientRect()\n  t.text(text)\n  while (Math.max(rect.width, rect.height) > space - tickLength) {\n    text = String(text)\n    text = text.slice(0, text.length - 1)\n    t.text(text + '...')\n    rect = t.node().getBoundingClientRect()\n    if (text.length == 0) break\n  }\n}\n\nexport function truncateString(string, space, font) {\n  var chars = space / font;\n  if (chars < string.length) {\n    return string.slice(0, Math.round(chars-5)) + '...'\n  } else {\n    return string\n  }\n}\n\n\n/**\n* Trys to use d3.selection to get element, if it doesnt exist, makes one\n* @param {d3.selection} sel selection in which to try and find object\n* @param {string} tag tag of which to try and select\n* @param {string} [cls=''] class of tag to try and grab\n* @returns {d3.selection} of either append or selected tag.cls within sel\n*/\nexport function safeSelect(sel, tag, cls) {\n  var clsStr = cls == undefined ? '' : '.'+cls;\n  var sSel = sel.select(tag+clsStr).empty()\n  ? sel.append(tag)\n  : sel.select(tag+clsStr)\n  return sSel\n  .classed(clsStr.replace('.', ''), true)\n  .attr('transform', sSel.attr('transform') == undefined ? 'translate(0,0)' : sSel.attr('transform'))\n}\n\n/**\n* evenly partitions the range [min, max] into n parts\n* @param {number} min\n* @param {number} max\n* @param {number} n\n* @returns {number[]} array of length n evenly partitioned between min and max\n*/\nexport function tickRange(min, max, n) {\n  var a = [min]\n  var d = max-min\n  var s = d / (n-1)\n  for (var i = 0; i < n-2; i++) { a.push(min + s * (i+1)) }\n  a.push(max)\n  return a\n}\n\n\nexport function euclideanDistance(p1, p2){\n  var a =  p1[0] - p2[0], b =  p1[1] - p2[1]\n  return Math.sqrt(a*a + b*b)\n}\n","import {uniqueElements} from './helpers';\n/*******************************************************************************\n**                                                                            **\n**                                                                            **\n**                              PROTOTYPES                                    **\n**                                                                            **\n**                                                                            **\n*******************************************************************************/\n/**\n* This function tests to see if all elements of the passed array are true.\n* @param {Array} array of values\n* @param {Function} [func function(value){return value == true;}] is applied to each value of the array and should return a boolean.\n* @returns {boolean} if all values are true by function\n*/\nexport function all( array, func ) {\n  if (func == undefined) { return array.every( function(value) { return value === true; }); }\n  return array.every( function(value) { return func(value); } );\n}\n\n/**\n* Counts the number of occurances of each element in the given array.\n* @param {Array} array of elements\n* @returns {Object} of key: value pairs where key is an element in the array and value is the number of times it occurs.\n*/\nexport function tally( array ) {\n  var tallies = {};\n  array.map( function ( element ) {\n    if ( hasQ(Object.keys(tallies), element) ) { tallies[element] = 1; }\n    else { tallies[element] += 1; }\n  });\n  return tallies;\n}\n/**\n* Short-hand for array.includes(item);\n* @param {Array} array\n* @param {*} item to test if contained in  {array}\n* @returns {boolean}\n*/\nexport function hasQ( array, item ) { return array.includes(item); }\n\n/**\n* Returns first item in array\n* @param {Array} array of items\n* @returns {*} array[0]\n*/\nexport function first( array ) { return array[0]; }\n\n/**\n* Returns last item in array\n* @param {Array} array of items\n* @returns {*} array[array.length-1]\n*/\nexport function last( array ) { return array[array.length-1]; }\n\n/**\n* Calculates the total value of numbers in passed array\n* @param {number[]} array of numerical values\n* @returns {number} sum over elements in array\n*/\nexport function total( array ) { return array.reduce((a, b) => a + b, 0) };\n\n/**\n* Removes duplicates in array\n* @param {Array} array of items\n* @returns {Array} of items such that item_i != item_j for all i < j\n* @see{@link uniqueElements} for the filtering function\n*/\nexport function unique( array ) { return array.filter( uniqueElements ); }\n\n/**\n* Filters passed array for specified indicies\n* @param {Array} array of items\n* @param {number[]} positions of integers such that i < array.length\n* @returns {Array} of items such that for any item_i, positions.includes(i) === true\n*/\nexport function get( array, positions ) {\n  return array.filter( function( value, index ) { return hasQ(positions, index); } );\n}\n\n/**\n* Determines if all elements in passed array are arrays themselves.\n* @param {Array} array of items\n* @returns {boolean} true if Array.isArray(e) is true for all e in array\n* @see{@link all}\n*/\nexport function listOfListsQ( array ) {\n  return all( array.map( function( element, index ) { return Array.isArray(element) } ) )\n}\n\n/**\n* Built on top of @see{@link get}, mapping if positions is a list of lists (@see{@link listOfListsQ})\n* @param {Array} array of items\n* @param {number[] | []number[] } positions of integers or list of positions of integers\n* @returns {boolean} returns specified positions from array. If nested positions passed, returns requested items in same structure.\n*/\nexport function cut( array, positions ) {\n  if ( listOfListsQ(array) ) { return positions.map(function(pos, i) { return array.get(pos); }); }\n  return get( array, positions );\n}\n\n/**\n* Given an array of objects, constructs new objects where each value is a list\n* based on the corresonding key, which is extracted by the parameter by\n* @param {Objects[]} array of objects\n* @param {string} by key within all objects of passed array\n* @param {string[]} [groups] saves some computation if all known values extracted by mapping over the parameter by are passed\n* @returns {Object} of key value pairs, where keys are all values of the key by from an object in the passed array and the value are those corresponding objects.\n*/\nexport function groupBy (array, by, groups) {\n  if (groups == undefined) {\n    groups = unique(array.map(function(elements, index){ return element[by]; }));\n    groups.map(function(value, index){groupped[value] = []})\n  }\n\n  var groupped = {};\n  array.map(function(element, index){groupped[element[by]].push(element)});\n  return groupped\n}\n\n/**\n* Tests if two arrays are equivalent\n* @param {Array} array\n* @param {Array} other\n* @returns {boolean} if every element of array matches that of other\n*/\nexport function arrayEquals(array, other) {\n  if (!other)\n      return false;\n  // compare lengths - can save a lot of time\n  if (array.length != other.length)\n      return false;\n\n  for (var i = 0, l=array.length; i < l; i++) {\n      // Check if we have nested arrays\n      if (array[i] instanceof Array && other[i] instanceof Array) {\n          // recurse into the nested arrays\n          if (!arrayEquals(array[i],other[i]))\n              return false;\n      }\n      else if (array[i] != other[i]) {\n          // Warning - two different object instances will never be equal: {x:20} != {x:20}\n          return false;\n      }\n  }\n  return true;\n}\n\n\n\n/**\n* Recursively tallies the number of elements at each level of the passed, putatively nested array\n* @param {Array} array of items which may include nested arrays\n* @param {number} [level=0] current depth in the recursion\n* @param {Array} [levelData=[]] keeps track of items seen so far at each depth\n* @returns {Array} stating the number of elements (array inclusive) found at each level of the array\n*/\nexport function elementsAtLevels(array, level, levelData) {\n  level = level == undefined ? 0 : level + 1;\n  levelData = levelData == undefined ? [] : levelData;\n  if ( level >= levelData.length ) { levelData.push(array.length)} else {levelData[level] += array.length }\n  array.map(function(e, i) {if (Array.isArray(e)){ elementsAtLevels(e, level, levelData) }})\n  return levelData\n}\n\n\n/**\n* Recursively tallies the number of elements of the passed, putatively nested array\n* @param {Array} array of items which may include nested arrays\n* @param {number} [elements=0] current number of elements seen so far\n* @returns {number} number of elements (array inclusive) found in passed array\n*/\nexport function numberOfElements( array, elements ) {\n  elements = elements == undefined ? 0 : elements;\n  array.map(function(e, i) {\n    if ( Array.isArray(e) ) { elements = numberOfElements(e, elements) }\n    else { elements += 1 }\n  })\n  return elements\n}\n\n/**\n* Concats all nested arrays in passed array to form a single array\n* @param {Array} array of putatively nested arrays\n* @param {Array} [flat=[]] current flattened array\n* @returns {Array} with every element in the same level\n*/\nexport function flatten( array, flat ) {\n  flat = flat == undefined ? [] : flat;\n  array.map(function(e, i){\n    if (Array.isArray(e)) {flat = flat.concat(flatten(e))}\n    else {flat.push(e)}\n  })\n  return flat;\n}\n\n/**\n* Search of list of lists to find which - if any - passed value is in\n* @param {Array[]} bins list of lists of values\n* @param {*} value item to test if in any of the bins\n* @returns {number} indicating the index of the bin in which value was found\n*/\nexport function whichBin(bins, value) {\n  var i = -1\n  for (var j = 0; j < bins.length; j++) { if (hasQ(bins[j],value)) {return j} }\n  return i\n}\n","import {hypenate, safeSelect} from './helpers';\nimport {total} from './array-functions';\n\n\n/**\n * calls console.group if d3sm.debugQ == true\n * @param {string} name of the group\n * @returns {undefined}\n */\nexport function consoleGroup(name) {\n  if (window.d3sm.debugQ === true){\n    console.group(name)\n  }\n}\n\n/**\n * calls console.groupEnd if d3sm.debugQ == true\n * @returns {undefined}\n */\nexport function consoleGroupEnd() {\n  if (window.d3sm.debugQ === true){\n    console.groupEnd()\n  }\n}\n\n/**\n * Calls console.log if d3sm.debugQ == true\n * @param {string} func name of the function logging\n * @param {string} msg to log\n * @param {Object} data to be logged along side the message\n * @returns {undefined}\n */\nexport function log(func, msg, data) {\n  if (window.d3sm.debugQ === true){\n    console.log(\n      `%c[d3sm::${func}]:\\t${msg}`,\n      [\n        'background: #6cd1ef',\n        'border-radius: 5000px',\n        'padding: 0px 2px',\n        'font-size: 14px'\n      ].join(';')\n    )\n    console.table(data)\n    // console.trace()\n  }\n}\n\n/**\n * Calls console.warn if d3sm.debugQ == true\n * @param {string} func name of the function warning\n * @param {string} msg to display\n * @param {Object} data to be displayed along side the message\n * @returns {undefined}\n */\nexport function warn(func, msg, data) {\n  if (window.d3sm.debugQ === true)\n    console.warn(\n      `%c[d3sm::${func}]:\\t${msg}`,\n      [\n        'background: #ffd53e',\n        'border-radius: 5000px',\n        'padding: 0px 2px',\n        'font-size: 14px'\n      ].join(';')\n    )\n    console.table(data)\n}\n/**\n * Calls the console.info if d3sm.debugQ == true\n * @param {string} func name of the function providing info\n * @param {string} msg to display\n * @param {Object} data to be displayed along side the message\n * @returns {undefined}\n */\nexport function info(func, msg, data) {\n  if (window.d3sm.debugQ)\n    console.info(\n      `%c[d3sm::${func}]:\\t${msg}`,\n      [\n        'background: #009ccd',\n        'border-radius: 5000px',\n        'padding: 0px 2px',\n        'font-size: 14px'\n      ].join(';')\n    )\n    console.table(data)\n}\n\n\n/**\n * Calls console.error if d3sm.debugQ == true\n * @param {string} func name of the function which sends the error\n * @param {string} msg to display\n * @param {Object} data to be displayed along side the message\n * @returns {undefined}\n */\nexport function error(func, msg, data) {\n  if (window.d3sm.debugQ)\n    console.error(`[d3sm::${func}]:\\t${msg}\\t%o`,data)\n}\n\n\n\n\n\n/**\n* Function for setting up containers for most plots with the y axis container\n* positioned on the left and the x axis container positioned on the bottom\n* @param {d3.selection} selection selection of container in which the svg is or should be made\n* @param {string} namespace namespace of the chart\n* @param {Object} [space={w:window.innerWidth, h:window.innerHeight}] the width (w) and height (h) availble\n* @param {number} [space.w=window.innerWidth] the available width in which to render the chart\n* @param {number} [space.h=window.innerHeight] the available height in which to render the chart\n\n* @param {Object} [margins={top: 0.01, bottom: 0.01, left: 0.01, right: 0.01}] the margins for the chart\n* @param {number} [margins.top=0.01] the top margin of the chart\n* @param {number} [margins.bottom=0.01] the bottom margin of the chart\n* @param {number} [margins.left=0.01] the left margin of the chart\n* @param {number} [margins.right=0.01] the right margin of the chart\n\n\n* @param {Object} [percentages = {axes:{x:0.1,y:0.1},space:{w:0.8,h:0.6}}] percentages of the paramater space of which to make the x and y axes as well as the percent of the availble space in which to render the plot\n* @param {Object} [percentages.axes={x:0.1,y:0.1}] the percentages of the paramater space, of which the x and y axes will take up\n* @param {number} [percentages.axes.xAxisPercent=0.1] the percentages of the paramater space, of which the x axis will take up\n* @param {number} [percentages.axes.yAxisPercent=0.1] the percentages of the paramater space, of which the y axis will take up\n\n* @param {Object} [percentages.space={w:0.8,h:0.6}] the percentages of the paramater space, of which the SVG's width and height will be set\n* @param {number} [percentages.space.percentOfSpaceForWidth=0.1] the percentages of the paramater space, of which the SVG's width will be set\n* @param {number} [percentages.space.percentOfSpaceForHeight=0.1] the percentages of the paramater space, of which the SVG's height will be set\n\n* @returns {Object} returns the selection and \"boundingRects\" of the plot container, x-axis container and y-axis container\n* as\n*\n* {\n*\n*   plot: {selection: plotSelection, rect: plotRect},\n*\n*   xAxis:{selection:xAxisSelection, rect:xAxisRect},\n*\n*   yAxis: {selection:yAxisSelection, rect:yAxisRect}\n*\n* }\n*\n* where each rect has form:\n*\n* {x: #, y: #, h: #, w: #}\n*\n* depicting the starting x and y coordinate of the coresponding container (also their default transform values) as well their height (h) ans width (w)\n*/\n// export function setupStandardChartContainers( selection, namespace, space, margins, percentages) {\n// export function setupStandardChartContainers(\n//   selection,\n//   namespace,\n//   space={w:availableWidth=window.innerWidth, h:availableHeight=window.innerHeight},\n//   margins={top:0.01, bottom:0.01, left:0.01, right:0.01},\n//   percentages={axes: {x: xAxisPercent=0.1, y: yAxisPercent=0.1}, space: {w: percentOfSpaceForWidth, h: percentOfSpaceForHeight}}\n// ) {\n//   if (space == undefined) { space = {w: window.innerWidth, h: window.innerHeight} }\n//   if (margins == undefined) { margins = {top: 0.01, bottom: 0.01, left: 0.01, right: 0.01} }\n//   if (percentages == undefined) { percentages = {}; }\n//   if (percentages.axes == undefined) { percentages.axes = { x:0.1, y:0.1 } }\n//   if (percentages.space == undefined) { percentages.space = { w: 0.8, h: 0.6 } }\n//\n//   // SVG width and height\n//   var svgSpace =  {\n//     w: space.w * percentages.space.w,\n//     h: space.h * percentages.space.h\n//   },\n//\n//   // Space after removing margins\n//   chartSpace = {\n//     w: svgSpace.w - (margins.left * space.w) - (margins.right * space.w),\n//     h: svgSpace.h - (margins.top * space.h) - (margins.bottom * space.h)\n//   },\n//\n//   // main dimension of x and y axies\n//   // e.g. defines how tall x axis is as length is determined by plotRect.w\n//   axesSpace = {\n//     x: chartSpace.h * percentages.axes.x,\n//     y: chartSpace.w * percentages.axes.y\n//   },\n//\n//   // space left for drawing the chart properly (e.g. bars, violins, etc)\n//   drawingSpace = {\n//     x: chartSpace.w - axesSpace.y,\n//     y: chartSpace.h - axesSpace.x\n//   },\n//\n//\n//   yAxisRect = {\n//     x: axesSpace.y + (margins.left * space.w),\n//     y: (margins.top * space.h),\n//     w: axesSpace.y,\n//     h: drawingSpace.y\n//   },\n//\n//   plotRect = {\n//     x: axesSpace.y + (margins.left * space.w),\n//     y: (margins.top * space.h),\n//     w: drawingSpace.x,\n//     h: drawingSpace.y\n//   },\n//\n//   xAxisRect = {\n//     x: axesSpace.y + (margins.left * space.w),\n//     y: (margins.top * space.h + plotRect.h),\n//     w: drawingSpace.x,\n//     h: axesSpace.x\n//   }\n//\n//\n//   var container = safeSelect(selection, 'svg', namespace)\n//     .style('width', svgSpace.w+'px')\n//     .style('height', svgSpace.h+'px')\n//\n//   var axes = safeSelect(container, 'g', hypenate(namespace, 'axes'))\n//\n//   // .attr('transform', \"translate(\"+plotRect.x+\",\"+plotRect.y+\")\"),\n//\n//   var plot = safeSelect(container, 'g', hypenate(namespace, 'plot'))\n//     .attr('transform', \"translate(\"+plotRect.x+\",\"+plotRect.y+\")\")\n//\n//   var xAxis = safeSelect(axes, 'g', hypenate(namespace, 'x-axis'))\n//     .attr('transform', \"translate(\"+xAxisRect.x+\",\"+xAxisRect.y+\")\")\n//\n//   var yAxis = safeSelect(axes, 'g', hypenate(namespace, 'y-axis'))\n//     .attr('transform', \"translate(\"+yAxisRect.x+\",\"+yAxisRect.y+\")\")\n//\n//   return {\n//     svg: {\n//       selection: container,\n//       rect: svgSpace\n//     },\n//     plot: {\n//       selection: plot,\n//       rect: plotRect\n//     },\n//     xAxis: {\n//       selection: xAxis,\n//       rect: xAxisRect\n//     },\n//     yAxis: {\n//       selection: yAxis,\n//       rect: yAxisRect\n//     }\n//   }\n//\n//   // return [plot, xAxis, yAxis]\n// }\n//\n//\n\n\n\n\n\nexport function setupStandardChartContainers(\n  selection,\n  namespace,\n  container,\n  margins={top:0.01, bottom:0.01, left:0.01, right:0.01},\n  svg={w:0.8, h:0.6}, // percent of container space for svg\n  axes={y:0.1, x:0.1}, // percent of container space for axes,\n  leg={x:0, margin:0, pos:'left'} // absolute width of legend and space on either size\n)\n{\n  if (container == undefined) {container = {w:window.innerWidth, h:window.Height}}\n  // SVG width and height\n\n  var svgSpace =  {\n    w: container.w * svg.w,\n    h: container.h * svg.h\n  }\n\n  var margPx = {\n    top: margins.top * svgSpace.h,\n    bottom: margins.bottom * svgSpace.h,\n    left: margins.left * svgSpace.w,\n    right: margins.right * svgSpace.w\n  },\n\n\n\n  // Space after removing margins\n  chartSpace = {\n    w: svgSpace.w - margPx.left - margPx.right,\n    h: svgSpace.h - margPx.top - margPx.bottom\n  },\n\n  // main dimension of x and y axies\n  // e.g. defines how tall x axis is as length is determined by plotRect.w\n  axesSpace = {\n    x: chartSpace.h * axes.x,\n    y: chartSpace.w * axes.y\n  },\n\n  // space left for drawing the chart properly (e.g. bars, violins, etc)\n  drawingSpace = {\n    x: chartSpace.w - axesSpace.y - leg.x - 2*leg.margin,\n    y: chartSpace.h - axesSpace.x\n  },\n\n\n  legRect = {\n    x: leg.margin + margPx.left + (leg.pos == 'left' ? 0 : drawingSpace.x + axesSpace.y),\n    y: margPx.top, // this is soomehow getting calculated incorectly\n    w: leg.x,\n    h: drawingSpace.y\n  },\n\n  yAxisRect = {\n    x: axesSpace.y + margPx.left + (leg.pos == 'left' ? leg.x + 2*leg.margin : 0),\n    y: margPx.top,\n    w: axesSpace.y,\n    h: drawingSpace.y\n  },\n\n  plotRect = {\n    x: axesSpace.y + margPx.left + (leg.pos == 'left' ? leg.x + 2*leg.margin : 0),\n    y: margPx.top,\n    w: drawingSpace.x,\n    h: drawingSpace.y\n  },\n\n  xAxisRect = {\n    x: axesSpace.y + margPx.left + (leg.pos == 'left' ? leg.x + 2*leg.margin : 0),\n    y: margPx.top + drawingSpace.y,\n    w: drawingSpace.x,\n    h: axesSpace.x\n  }\n\n\n\n  container = d3sm.safeSelect(selection, 'svg', namespace)\n    .style('width', svgSpace.w+'px')\n    .style('height', svgSpace.h+'px')\n\n  var axes = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'axes'))\n\n  var leg = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'legend'))\n  .attr('transform', \"translate(\"+legRect.x+\",\"+legRect.y+\")\")\n\n  var plot = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'plot'))\n    .attr('transform', \"translate(\"+plotRect.x+\",\"+plotRect.y+\")\")\n\n  var xAxis = d3sm.safeSelect(axes, 'g', d3sm.hypenate(namespace, 'x-axis'))\n    .attr('transform', \"translate(\"+xAxisRect.x+\",\"+xAxisRect.y+\")\")\n\n  var yAxis = d3sm.safeSelect(axes, 'g', d3sm.hypenate(namespace, 'y-axis'))\n    .attr('transform', \"translate(\"+yAxisRect.x+\",\"+yAxisRect.y+\")\")\n\n  return {\n    svg: {\n      selection: container,\n      rect: svgSpace\n    },\n    plot: {\n      selection: plot,\n      rect: plotRect\n    },\n    xAxis: {\n      selection: xAxis,\n      rect: xAxisRect\n    },\n    yAxis: {\n      selection: yAxis,\n      rect: yAxisRect\n    },\n    legend: {\n      selection: leg,\n      rect: legRect\n    }\n  }\n}\n\n\n\n/**\n* Adds a clip-path rect and binds it to container\n* @param {d3.selection} container in which to add the clip-path and to which to bind the cliping path to\n* @param {Object} rect the coordinates (x, y, width, height) of the clip-path\n* @param {string} namespace\n* @returns {d3.selection} of the clip-path rect\n*/\nexport function cpRect(container, rect, namespace) {\n  var defs = safeSelect(container, 'defs', hypenate(namespace, 'definitions'))\n  var cp = safeSelect(defs, 'clipPath', hypenate(namespace, 'clip-path'))\n  .attr('id', hypenate(namespace, 'clip-path'))\n\n  var cpRect = safeSelect(cp, 'rect')\n  .attr('x', rect.x)\n  .attr('y', rect.y)\n  .attr('width', rect.width)\n  .attr('height', rect.height)\n\n  defs.raise()\n  // set clipping path to container\n  container.attr('clip-path', 'url(#'+ hypenate(namespace, 'clip-path')+')')\n\n  return cpRect\n}\n\n\n/**\n* Adds a background rect t to container\n* @param {d3.selection} container in which to add the background rectangle\n* @param {Object} rect the coordinates (x, y, width, height) of the background\n* @param {string} fill the color of the background\n* @returns {d3.selection} of the background fill\n*/\nexport function bgRect(container, rect, fill) {\n  return safeSelect(container, 'rect', 'bg')\n  .attr('x', rect.x)\n  .attr('y', rect.y)\n  .attr('width', rect.width)\n  .attr('height', rect.height)\n  .attr('fill', fill)\n}\n\n\n/**\n* Sets up the container for making chart elements. This includes making\n* a clip-path rect bound to the passed container, a background rect, and\n* a g element with class <namespace>-object-container.\n* @param {d3.selection} container in which to add the clip-path and background\n* @param {string} namespace\n* @param {Object} rect the coordinates (x, y, width, height) of the background and clip-path\n* @param {string} fill the color of the background\n* @returns {d3.selection} of g.<namespace>-object-container\n*\n* @see{@link bgRect}\n* @see{@link cpRect}\n*/\nexport function setupContainer(selection, namespace, rect, fill) {\n  // the container for three main items, bg, defs, and object-container\n  var\n  container = safeSelect(selection, 'g', namespace),\n  bg = bgRect(container, rect, fill),\n  cp = cpRect(container, rect, namespace),\n  objectContainer = safeSelect(container, 'g', hypenate(namespace, 'object-container'))\n  return objectContainer\n}\n\n\n/**\n* determines the width of an object for the calling plotting function\n* @param {number} freeSpace how much space is avalible\n* @param {number} numberOfObjects how many object do we need\n* @param {number} minObjectWidth how small are these objects allowed to be\n* @param {number} maxObjectWidth how large are these object allowed to be\n* @param {number} sizeOfSpacer percent of freeSpace that a single spacer should take up (need numberOfObjects - 1 spacers)\n* @param {boolean} overflowQ can we go beyond alloted space\n* @returns {number} how large object should be\n* function tries to keep object within min / max width, but wil default to\n* 5e-10 (smallest consistenly visible by svg size of element) if overflowQ is false\n*/\nexport function calculateWidthOfObject(freeSpace, numberOfObjects, minObjectWidth, maxObjectWidth, sizeOfSpacer, overflowQ) {\n  var sizeOfSpacer =\n  sizeOfSpacer == 0 || sizeOfSpacer > 1\n  ? sizeOfSpacer\n  : freeSpace * sizeOfSpacer\n\n  var numberOfSpacers = numberOfObjects - 1\n  var spaceTakenBySpacers = numberOfSpacers * sizeOfSpacer\n  var remainingSpace = freeSpace - spaceTakenBySpacers\n  remainingSpace = remainingSpace < 0 ? 0 : remainingSpace\n  var objectWidth = remainingSpace / numberOfObjects\n\n  if ( overflowQ && minObjectWidth != undefined && objectWidth < minObjectWidth ) { objectWidth = minObjectWidth }\n  // if ( maxObjectWidth != undefined && objectWidth > maxObjectWidth ) { objectWidth = maxObjectWidth }\n  if ( overflowQ && maxObjectWidth != undefined && objectWidth < maxObjectWidth ) { objectWidth = maxObjectWidth }\n  return Math.max(objectWidth, 5e-10)\n}\n\n/**\n* @param {Array[]} data list data (can be nested). If nested will create more complex spacer size\n* @param {number} freeSpace how much space is avalible\n* @param {number} objectWidth @see{@link calculateWidthOfObject}\n* @param {number} numberOfObjects how many object do we need\n* @param {number} baseSpacerSize percent of freeSpace that a single spacer should take up (need numberOfObjects - 1 spacers)\n* @param {boolean} overflowQ can we go beyond alloted space\n* @returns {number} returns size that spacer should be at level=0\n*/\nexport function calculateWidthOfSpacer(data, freeSpace, objectWidth, numberOfObjects, baseSpacerSize, overflowQ) {\n  if (overflowQ) {\n    // var limitedNumberOfObjects = numberOfObjects > 6 ? 6 : numberOfObjects\n    // var spaceLeft = freeSpace - limitedNumberOfObjects * objectWidth\n    // return spaceLeft / (limitedNumberOfObjects - 1)\n    return freeSpace * baseSpacerSize\n  }\n  var spacersAtEachLevel = spacersNeededAtEachLevel(data)\n  var totalSpacerPercent = total(spacersAtEachLevel.map(function(e, i) {return e * 1 / (i+1)}))\n  var baseSpacerSize = (freeSpace - (objectWidth * numberOfObjects)) / totalSpacerPercent\n  // console.log(freeSpace, objectWidth, numberOfObjects, totalSpacerPercent)\n  // console.log(totalSpacerPercent, baseSpacerSize, totalSpacerPercent * baseSpacerSize)\n  return isNaN(baseSpacerSize) ? 0 : baseSpacerSize\n}\n\n\n/**\n* Calculates number of spacers needed to seperate elements at each level.\n* @param {Array[]} array list data (can be nested). If nested will create more complex spacer size\n* @param {number} [level=0] current level, used in recusrion\n* @param {Array} [levelData=[]] how many spacers needed at a given level\n* @returns {Array} levelData\n*\n* @example\n* array = [[1,2], [3,4]]\n* // returns [1, 2]\n* as at level=0 the only spacer needed is between [1,2] and [3,4]\n* and at level=1 the only two spacers needed is between 1 and 2 as well as\n* 3 and 4 since the spacer between 2 and 3 is handled at level=0\n*/\nexport function spacersNeededAtEachLevel (array, level, levelData ) {\n  if ( level == undefined ) { level = 0;  } else { level += 1 }\n  if ( levelData == undefined ) { levelData = []; }\n  if ( level >= levelData.length ) { levelData.push(array.length - 1) }\n  else { levelData[level] += array.length - 1 }\n  array.map(function(e, i) { if (Array.isArray(e)) { spacersNeededAtEachLevel(e, level, levelData) } } )\n  return levelData\n}\n\n\n\n\n/**\n* Draws a whisker for @see{@link boxwhisker}\n* @param {boolean} dir direction to draw whisker, should be either true (up, top) or false (down or bottom)\n* @param {number} x starting x coordinate in which to draw whisker\n* @param {number} y starting y coordinate in which to draw whisker\n* @param {number} w width of space in which to draw whisker\n* @param {number} h height of space in which to draw whisker\n* @param {number} per percentage of w or h (depends on o) to make whisker\n* @param {boolean} o orientation, true is horizontal and false is vertical\n* @returns {string} representing the svg path (i.e. the d attribute for a path tag)\n*/\nexport function whiskerPath(dir, x, y, w, h, per, o) {\n  // d = direction (true is up), p = percent width\n  if (dir == 'up' || dir == 'top' || dir == true) {dir = true}\n  if (dir == 'down' || dir == 'bottom' || dir == false) {dir = false}\n  o = o == undefined ? 'horizontal' : o\n  per = per == undefined ? 1 : per\n  if (o != \"horizontal\") {\n    var hh = h * per ,\n    w = dir ? w : -w ,\n    a = dir ? x + w : x ,\n    b = dir ? x : x + w ,\n    c = dir ? a : b\n    p = \"M \" + a + ' ' + (     h / 2      ) + ' '\n      + 'L ' + b + ' ' + (     h / 2      ) + ' '\n      + 'M ' + c + ' ' + ( h / 2 - hh / 2 ) + ' '\n      + 'L ' + c + ' ' + ( h / 2 + hh / 2 ) + ' '\n\n    return p\n  }\n  var ww = w * per,\n  a = dir ? y + h : y  ,\n  b = dir ? y : y + h  ,\n  p = \"M \" + (  w / 2  ) + ' ' + a + ' ' // straight line part\n    + 'L ' + (  w / 2  ) + ' ' + b + ' ' // straight line part\n    + 'h ' + ( -ww / 2 ) + ' ' + 0 + ' ' // horizontal line part\n    + 'h ' + (    ww   ) + ' ' + 0 + ' '\n  return p\n}\n\n\n\n\n\n\n\n\n\n\n\n\n\nexport function resizeDebounce(f, wait) {\n  var resize = debounce(function(){f()},wait)\n  window.addEventListener('resize', resize)\n}\n\n\n\nfunction debounce(func, wait, immediate) {\n  var timeout;\n    return function() {\n        var context = this, args = arguments;\n        var later = function() {\n            timeout = null;\n            if (!immediate) func.apply(context, args);\n        };\n        var callNow = immediate && !timeout;\n        clearTimeout(timeout);\n        timeout = setTimeout(later, wait);\n        if (callNow) func.apply(context, args);\n    };\n}\n","import {log, warn, error, info} from './utils';\n/*******************************************************************************\n**                                                                            **\n**                                                                            **\n**                              SPACEGROUPING                                 **\n**                                                                            **\n**                                                                            **\n*******************************************************************************/\n/**\n * Produces a function for spacing objects by an arbitrarly complex grouping\n * @returns {recursivelyPosition} the function for moving the objects\n * (see {@link groupingSpacer#recursivelyPosition})\n * @namespace groupingSpacer\n */\nexport function groupingSpacer() {\n  var\n  /*@var {boolean} horizontalQ @default*/\n\n  /**\n  * Whether or not to space objects horizontally or vertically.\n  * (see {@link groupingSpacer.horizontalQ})\n  * @param {boolean} [horizontalQ=true]\n  * @memberof groupingSpacer#\n  * @instance\n  */\n  horizontalQ = true,\n  /**\n  * The scale to use to position elements if {@link groupingSpacer#moveby}=\"string\"\n  * (see {@link groupingSpacer.scale})\n  * @param {d3.scale} [scale=d3.scaleLinear()]\n  * @memberof groupingSpacer#\n  * @instance\n  */\n  scale = d3.scaleLinear(),\n  /**\n  * How elements in the complex grouping should be moved over by.\n  * By default, moveby=\"category\", which moves objects by the complex grouping\n  * But objects can also be moved over by scale.\n  * (see {@link groupingSpacer.moveby})\n  * @param {string} [moveby=\"category\"]\n  * @memberof groupingSpacer#\n  * @instance\n  */\n  moveby = 'category',\n  /**\n  * How many objects are there in total\n  * (see {@link groupingSpacer.numberOfObjects})\n  * @param {number} [numberOfObjects=none]\n  * @memberof groupingSpacer#\n  * @instance\n  */\n  numberOfObjects,\n  /**\n  * The class given to an nested <g> tag whose parent(s) have the correct transition\n  * properties\n  * (see {@link groupingSpacer.numberOfObjects})\n  * @param {string} [numberOfObjects='d3sm-groupped-item']\n  * @memberof groupingSpacer#\n  * @instance\n  */\n  objectClass = 'd3sm-groupped-item',\n  /**\n  * The size of the objects being positioned\n  * (see {@link groupingSpacer.objectSize})\n  * @param {number} [objectSize=none]\n  * @memberof groupingSpacer#\n  * @instance\n  */\n  objectSize,\n  /**\n  * The size of the un-nested spacer between objects\n  * (see {@link groupingSpacer.spacerSize})\n  * @param {number} [spacerSize=none]\n  * @memberof groupingSpacer#\n  * @instance\n  */\n  spacerSize,\n  /**\n  * The duration of transitions in ms\n  * (see {@link groupingSpacer.transitionDuration})\n  * @param {number} [transitionDuration=1000]\n  * @memberof groupingSpacer#\n  * @instance\n  */\n  transitionDuration = 1000,\n  /**\n  * The ease function for the transitions\n  * (see {@link groupingSpacer.easeFunc})\n  * @param {d3.ease} [easeFunc=d3.easeSin]\n  * @memberof groupingSpacer#\n  * @instance\n  */\n  easeFunc = d3.easeSin,\n  /**\n  * The namespace for the objects being moved\n  * (see {@link groupingSpacer.namespace})\n  * @param {string} [namespace='spacer']\n  * @memberof groupingSpacer#\n  * @instance\n  */\n  namespace = 'spacer',\n  /**\n  * The animation for new objects being added\n  * (see {@link groupingSpacer.enterFunction})\n  * @param {function} enterFunction\n  * @memberof groupingSpacer#\n  * @instance\n  * @example\n  * // by default\n  * function(newObjectSelection) {\n  *  newObjectSelection.attr('transform', function(d, i){\n  *    var\n  *    x = horizontalQ ? objectSize * numberOfObjects + spacerSize * (numberOfObjects - 1) : 0,\n  *    y = !horizontalQ ? objectSize * numberOfObjects + spacerSize * (numberOfObjects - 1) : 0,\n  *    t = 'translate('+x+','+y+')'\n  *    return t\n  *  })\n  * }\n  */\n  enterFunction = function(cur) {\n    cur.attr('transform', function(d, i){\n      var\n      // x = horizontalQ ? objectSize * numberOfObjects + spacerSize * (numberOfObjects - 1) : 0,\n      // y = !horizontalQ ? objectSize * numberOfObjects + spacerSize * (numberOfObjects - 1) : 0,\n      x = horizontalQ ? window.outerWidth : 0,\n      y = !horizontalQ ? window.outerWidth : 0,\n      t = 'translate('+x+','+y+')'\n      // if(y == undefined) {console.log(cur.node(), y, d)}\n      return t\n    })\n  },\n  /**\n  * The animation for old objects being removed\n  * (see {@link groupingSpacer.exitFunction})\n  * @param {function} exitFunction\n  * @memberof groupingSpacer#\n  * @instance\n  * @example\n  * // by default\n  * oldObjectSelection.transition().duration(transitionDuration).ease(easeFunc)\n  * .attr('transform', function(d, i){\n  *     var\n  *   x = horizontalQ ? objectSize * numberOfObjects + spacerSize * (numberOfObjects - 1) : 0,\n  *   y = !horizontalQ ? objectSize * numberOfObjects + spacerSize * (numberOfObjects - 1) : 0,\n  *   t = 'translate('+x+','+y+')'\n  *   return t\n  * }).remove()\n  */\n  exitFunction = function(cur){\n    log(\"groupingSpacer\", \"exiting with\", {current: cur, currentNode: cur.node()})\n    cur.selectAll('g').classed('to-remove', true)\n\n    cur.transition().duration(transitionDuration*0.9).ease(easeFunc)\n    .attr('transform', function(d, i){\n      var\n      // x = horizontalQ ? objectSize * numberOfObjects + spacerSize * (numberOfObjects - 1) : 0,\n      // y = !horizontalQ ? objectSize * numberOfObjects + spacerSize * (numberOfObjects - 1) : 0,\n      x = horizontalQ ? window.outerWidth : 0,\n      y = !horizontalQ ? window.outerWidth : 0,\n      t = 'translate('+x+','+y+')'\n      // if(y == undefined) {console.log(cur.node(), y, d)}\n      return t\n    }).remove()\n  }\n\n  /**\n   * Gets / sets horizontalQ (whether or not to space objects horizontally or vertically).\n   * (see {@link groupingSpacer#horizontalQ})\n   * @param {string} [_=none]\n   * @returns {groupingSpacer | string}\n   * @memberof groupingSpacer\n   * @static\n   */\n  recursivelyPosition.horizontalQ = function(_) { return arguments.length ? (horizontalQ = _, recursivelyPosition) : horizontalQ }\n  /**\n   * Gets / sets the scale to use to position elements if {@link groupingSpacer#moveby}=\"string\"\n   * (see {@link groupingSpacer#scale})\n   * @param {d3.scale} [_=none]\n   * @returns {groupingSpacer | d3.scale}\n   * @memberof groupingSpacer\n   * @static\n   */\n  recursivelyPosition.scale = function(_) { return arguments.length ? (scale = _, recursivelyPosition) : scale }\n  /**\n   * Gets / sets moveby (whether or not to move by scale or by grouping).\n   * (see {@link groupingSpacer#moveby})\n   * @param {string} [_=none]\n   * @returns {groupingSpacer | string}\n   * @memberof groupingSpacer\n   * @static\n   */\n  recursivelyPosition.moveby = function(_) { return arguments.length ? (moveby = _, recursivelyPosition) : moveby }\n  /**\n   * Gets / sets numberOfObjects.\n   * (see {@link groupingSpacer#numberOfObjects})\n   * @param {number} [_=none]\n   * @returns {groupingSpacer | number}\n   * @memberof groupingSpacer\n   * @static\n   */\n  recursivelyPosition.numberOfObjects = function(_) { return arguments.length ? (numberOfObjects = _, recursivelyPosition) : numberOfObjects }\n  /**\n   * Gets / sets the objectClass (will be applied to <g> elements).\n   * (see {@link groupingSpacer#objectClass})\n   * @param {string} [_=none]\n   * @returns {groupingSpacer | string}\n   * @memberof groupingSpacer\n   * @static\n   */\n  recursivelyPosition.objectClass = function(_) { return arguments.length ? (objectClass = _, recursivelyPosition) : objectClass }\n  /**\n   * Gets / sets the objectSize.\n   * (see {@link groupingSpacer#objectSize})\n   * @param {number} [_=none]\n   * @returns {groupingSpacer | number}\n   * @memberof groupingSpacer\n   * @static\n   */\n  recursivelyPosition.objectSize = function(_) { return arguments.length ? (objectSize = _, recursivelyPosition) : objectSize }\n  /**\n   * Gets / sets the spacerSize.\n   * (see {@link groupingSpacer#spacerSize})\n   * @param {number} [_=none]\n   * @returns {groupingSpacer | number}\n   * @memberof groupingSpacer\n   * @static\n   */\n  recursivelyPosition.spacerSize = function(_) { return arguments.length ? (spacerSize = _, recursivelyPosition) : spacerSize }\n  /**\n   * Gets / sets the transitionDuration.\n   * (see {@link groupingSpacer#transitionDuration})\n   * @param {number} [_=none]\n   * @returns {groupingSpacer | number}\n   * @memberof groupingSpacer\n   * @static\n   */\n  recursivelyPosition.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, recursivelyPosition) : transitionDuration }\n  /**\n   * Gets / sets the easeFunc.\n   * (see {@link groupingSpacer#easeFunc})\n   * @param {d3.ease} [_=none]\n   * @returns {groupingSpacer | d3.ease}\n   * @memberof groupingSpacer\n   * @static\n   */\n  recursivelyPosition.easeFunc = function(_) { return arguments.length ? (easeFunc = _, recursivelyPosition) : easeFunc }\n  /**\n   * Gets / sets the namespace.\n   * (see {@link groupingSpacer#namespace})\n   * @param {string} [_=none]\n   * @returns {groupingSpacer | string}\n   * @memberof groupingSpacer\n   * @static\n   */\n  recursivelyPosition.namespace = function(_) { return arguments.length ? (namespace = _, recursivelyPosition) : namespace }\n  /**\n   * Gets / sets the enterFunction.\n   * (see {@link groupingSpacer#enterFunction})\n   * @param {function} [_=none]\n   * @returns {groupingSpacer | function}\n   * @memberof groupingSpacer\n   * @static\n   */\n  recursivelyPosition.enterFunction = function(_) { return arguments.length ? (enterFunction = _, recursivelyPosition) : enterFunction }\n  /**\n   * Gets / sets the exitFunction.\n   * (see {@link groupingSpacer#exitFunction})\n   * @param {function} [_=none]\n   * @returns {groupingSpacer | function}\n   * @memberof groupingSpacer\n   * @static\n   */\n  recursivelyPosition.exitFunction = function(_) { return arguments.length ? (exitFunction = _, recursivelyPosition) : exitFunction }\n\n\n  /**\n   * recursively position the objects inside of the selection.\n   * @param {d3.selection} selection\n   * @param {Object} data\n   * @param {level} [level=0] recursion depth\n   * @returns {number} (how much to move next element)\n   * @memberof groupingSpacer#\n   */\n  function recursivelyPosition(selection, data, level) {\n    if ( level == undefined ) { level = 0;  }\n\n    var currentSelection = selection.selectAll('g.'+namespace+'[level=\"'+level+'\"]').data(data)\n    var enter = currentSelection.enter().append('g').attr('level', level).attr('class', namespace)\n    var exit = currentSelection.exit()\n    currentSelection = currentSelection.merge(enter)\n\n\n    if (typeof exitFunction == 'function' ){ exit.each(function(d, i){ exitFunction(d3.select(this))}) }\n    else{exit.remove()}\n    // spacer for current level\n    var levelSpacer = spacerSize / (level+1)\n    // movement for current level\n    var move = 0\n    currentSelection.each(function(currentElement, index) {\n      var t = d3.select(this)\n      if (t.attr('transform') == undefined && typeof enterFunction == 'function') { enterFunction(t) }\n\n      t.transition().duration(transitionDuration).ease(easeFunc)\n      .attr('transform', function(d, i) {\n        var\n        x = horizontalQ ? (moveby ==\"scale\" ? scale(d) : move) : 0,\n        y = !horizontalQ ? (moveby ==\"scale\" ? scale(d) : move): 0,\n        t = 'translate('+x+','+y+')'\n        return t\n      })\n\n      if (Array.isArray(currentElement)) {\n        move += recursivelyPosition(t, currentElement, level+1)\n        var toRemove = t.selectAll('g.'+namespace+'[level=\"'+(level)+'\"] > g.'+objectClass+'.'+namespace)\n        if (typeof exitFunction == 'function' ){ toRemove.each(function(d, i){ exitFunction(d3.select(this))}) }\n        else{toRemove.remove()}\n      }\n      else {\n        move += objectSize\n        var obj = t.select('g.'+namespace+'[level=\"'+level+'\"] > g.'+objectClass+'.'+namespace)\n        if (obj.empty()) { obj = t.append('g').attr('class', objectClass).classed(namespace, true) }\n        obj.attr('parent-index', index)\n        var toRemove = t.selectAll('g.'+namespace+'[level=\"'+(level+1)+'\"]')\n\n        if (typeof exitFunction == 'function' ){ toRemove.each(function(d, i){ exitFunction(d3.select(this))}) }\n        else{toRemove.remove()}\n      }\n      move += (index == currentSelection.size()-1) ? 0 : levelSpacer\n    })\n    return move\n  }\n  return recursivelyPosition\n}\n","import {modifyHexidecimalColorLuminance} from './helpers';\n\n/**\n * Creates a colorFunction\n * @constructor colorFunction\n * @namespace colorFunction\n * @returns {function} colorFunction\n */\nexport function colorFunction() {\n  var\n  data,\n\n  /**\n  * Default colors to use\n  * @param {number[]} [colors=[\"#2c7bb6\", \"#00a6ca\", \"#00ccbc\", \"#90eb9d\", \"#ffff8c\", \"#f9d057\", \"#f29e2e\", \"#e76818\", \"#d7191c\"]]\n  * @memberof colorFunction#\n  * @property\n  */\n  colors = [\"#2c7bb6\", \"#00a6ca\", \"#00ccbc\", \"#90eb9d\", \"#ffff8c\", \"#f9d057\", \"#f29e2e\", \"#e76818\", \"#d7191c\"],\n  /**\n  * Interpolator for colors\n  * @param {d3.interpolation} [interpolation=d3.interpolateRgb]\n  * @memberof colorFunction#\n  * @property\n  */\n  interpolation = d3.interpolateRgb,\n  /**\n  * Function for modifying color luminance\n  * @param {function} [modifyOpacity=modifyHexidecimalColorLuminance]\n  * @memberof colorFunction#\n  * @property\n  */\n  modifyOpacity = modifyHexidecimalColorLuminance,\n  /**\n  * How to modify color for stroke\n  * @param {number} [strokeOpacity=0]\n  * @memberof colorFunction#\n  * @property\n  */\n  strokeOpacity = 0,\n  /**\n  * How to modify color for fill\n  * @param {number} [fillOpacity=0.4]\n  * @memberof colorFunction#\n  * @property\n  */\n  fillOpacity = 0.4,\n  /**\n  * How to determine the color to use\n  * @param {string} [colorBy='index']\n  * @memberof colorFunction#\n  * @property\n  */\n  colorBy = 'index',\n  /**\n  * Sets the scale for interpolating the colors\n  * @param {number[]} [dataExtent=[0, colors.length - 1]]\n  * @memberof colorFunction#\n  * @property\n  */\n  dataExtent = [0, colors.length - 1],\n  /**\n  * Extracts the value to color by\n  * @param {function} [valueExtractor=function(k, v, i) {return v}]\n  * @memberof colorFunction#\n  * @property\n  */\n  valueExtractor = function(k, v, i) {return v},\n\n  /**\n  * Extracts the category to color by\n  * @param {function} [categoryExtractor=function(k, v, i) {return v.category}]\n  * @memberof colorFunction#\n  * @property\n  */\n  categoryExtractor = function(k, v, i) {return v.category},\n\n  /**\n  * The different type of categories of which to color by\n  * @param {string[]} [categories=undefined]\n  * @memberof colorFunction#\n  * @property\n  */\n  categories,\n\n  /**\n  * Scale for interpolating the colors\n  * @param {d3.scale} [scale=d3.scaleLinear()]\n  * @memberof colorFunction#\n  * @property\n  */\n  scale = d3.scaleLinear()\n  .interpolate(interpolation).domain(dataExtent).range(colors),\n  helperScale = d3.scaleLinear()\n\n\n\n  // var h = x => '#' + x.match(/\\d+/g).map(y = z => ((+z < 16)?'0':'') + (+z).toString(16)).join('');\n  var h = function(x) {\n    return \"#\" + x.match(/\\d+/g).map(\n      function(y, i) {\n        return  ((+y < 16)?'0':'') + (+y).toString(16)\n      }).join('');\n  }\n\n  /**\n   * Gets or sets the default colors\n   * (see {@link colorFunction#colors})\n   * @param {number[]} [_=none]\n   * @returns {colorFunction | number[]}\n   * @memberof colorFunction\n   * @property\n   */\n  colorFunction.colors = function(_) {\n    return arguments.length\n    ?\n      (\n        colors = _,\n        scale.range(colors),\n        colorFunction\n      )\n    : colors;\n  };\n  /**\n   * Gets or sets the function for interpolating the colors\n   * (see {@link colorFunction#interpolation})\n   * @param {d3.interpolation} [_=none]\n   * @returns {colorFunction | d3.interpolation}\n   * @memberof colorFunction\n   * @property\n   */\n  colorFunction.interpolation = function(_) {\n    return arguments.length\n    ?\n    (\n      interpolation = _,\n      scale.interpolate(interpolation).range(colors),\n      colorFunction\n    )\n    : interpolation;\n  };\n  /**\n   * Gets or sets the values for the scale which transforms the value to a color\n   * (see {@link colorFunction#dataExtent})\n   * @param {number[]} [_=none]\n   * @returns {colorFunction | number[]}\n   * @memberof colorFunction\n   * @property\n   */\n  colorFunction.dataExtent = function(_) {\n    return arguments.length\n    ? (\n        dataExtent = _,\n        scale.domain(dataExtent).interpolate(scale.interpolate()),\n        colorFunction\n      )\n    : dataExtent;\n  };\n  /**\n   * Gets or sets the vthe scale which transforms the value to a color\n   * (see {@link colorFunction#scale})\n   * @param {d3.scale} [_=none]\n   * @returns {colorFunction | d3.scale}\n   * @memberof colorFunction\n   * @property\n   */\n  colorFunction.scale = function(_) {\n    return arguments.length\n    ? (\n        _ = _.domain(scale.domain()).interpolate(scale.interpolate()).range(scale.range()),\n        scale = _,\n        colorFunction\n      )\n    : scale;\n  };\n  /**\n   * Gets or sets the function for modify opacity\n   * (see {@link colorFunction#modifyOpacity})\n   * @param {function} [_=none]\n   * @returns {colorFunction | function}\n   * @memberof colorFunction\n   * @property\n   */\n  colorFunction.modifyOpacity = function(_) { return arguments.length ? (modifyOpacity = _, colorFunction) : modifyOpacity; };\n  /**\n   * Gets or sets the value to modify the color for the stroke via {@link colorFunction#modifyOpacity}\n   * (see {@link colorFunction#strokeOpacity})\n   * @param {number} [_=none]\n   * @returns {colorFunction | number}\n   * @memberof colorFunction\n   * @property\n   */\n  colorFunction.strokeOpacity = function(_) { return arguments.length ? (strokeOpacity = _, colorFunction) : strokeOpacity; };\n  /**\n   * Gets or sets the value to modify the color for the stroke via {@link colorFunction#fillOpacity}\n   * (see {@link colorFunction#fillOpacity})\n   * @param {number} [_=none]\n   * @returns {colorFunction | number}\n   * @memberof colorFunction\n   * @property\n   */\n  colorFunction.fillOpacity = function(_) { return arguments.length ? (fillOpacity = _, colorFunction) : fillOpacity; };\n  /**\n   * Gets or sets the value to colorBy\n   * (see {@link colorFunction#colorBy})\n   * @param {string} [_=none]\n   * @returns {colorFunction | string}\n   * @memberof colorFunction\n   * @property\n   */\n  colorFunction.colorBy = function(_) { return arguments.length ? (colorBy = _, colorFunction) : colorBy; };\n  /**\n   * Gets or sets the value of valueExtractor\n   * (see {@link colorFunction#valueExtractor})\n   * @param {function} [_=none]\n   * @returns {colorFunction | function}\n   * @memberof colorFunction\n   * @property\n   */\n  colorFunction.valueExtractor = function(_) { return arguments.length ? (valueExtractor = _, colorFunction) : valueExtractor; };\n\n  /**\n   * Gets or sets the value of categoryExtractor\n   * (see {@link colorFunction#categoryExtractor})\n   * @param {function} [_=none]\n   * @returns {colorFunction | function}\n   * @memberof colorFunction\n   * @property\n   */\n  colorFunction.categoryExtractor = function(_) { return arguments.length ? (categoryExtractor = _, colorFunction) : categoryExtractor; };\n  /**\n   * Gets or sets the value of categoryExtractor\n   * (see {@link colorFunction#categories})\n   * @param {string[]} [_=none]\n   * @returns {colorFunction | string[]}\n   * @memberof colorFunction\n   * @property\n   */\n  colorFunction.categories = function(_) { return arguments.length ? (categories = _, colorFunction) : categories; };\n\n\n  function colorFunction(key, value, index, type, hoverQ) {\n    var c,\n    opac = type == \"fill\" ? fillOpacity : strokeOpacity;\n\n\n    updateScale()\n\n    if (colorBy == \"index\") {\n      c = (type != undefined) ? modifyOpacity(h(scale(index)), opac) : h(scale(index))\n    }\n\n    else if (colorBy == 'value') {\n      var v = valueExtractor(key, value, index);\n      // if (v < dataExtent[0]) {dataExtent[0] = v; updateScale()}\n      // if (v > dataExtent[1]) {dataExtent[1] = v; updateScale()}\n\n      c = (type != undefined) ? modifyOpacity(h(scale(v)), opac) : h(scale(v))\n    }\n\n    else if (colorBy == 'category' ){\n      var cat = categoryExtractor(key, value, index);\n      var v = categories.indexOf(cat)\n      c = (type != undefined) ? modifyOpacity(h(scale(v)), opac) : h(scale(v))\n\n    }\n\n    else {\n      c = (type != undefined) ? modifyOpacity(h(scale(index)), opac) : h(scale(index))\n    }\n\n    return c\n  }\n\n  function updateScale(){\n\n\n    helperScale.domain([0, colors.length])\n    if (colorBy == 'category' && categories != undefined) { helperScale.range([0, categories.length]) }\n    else { helperScale.range(dataExtent) }\n\n\n    var a = Array(colors.length).fill(0).map(function(d, i){ return helperScale(i) })\n    scale.domain(a)\n  }\n\n  return colorFunction\n}\n","import {safeSelect, round} from './helpers';\nimport {log, warn, info, error, consoleGroup, consoleGroupEnd} from './utils';\n/*******************************************************************************\n**                                                                            **\n**                                                                            **\n**                                 TOOLTIP                                    **\n**                                                                            **\n**                                                                            **\n*******************************************************************************/\n/**\n * Produces a function for handling the tooltip\n *\n * {@link https://sumneuron.gitlab.io/d3sm/demos/tooltip-design/index.html Demo}\n * @param {d3.selection} selection\n * @returns {tooltip}\n * @namespace tooltip\n */\nexport function tooltip( selection ) {\n\n  var\n  keys,\n  values,\n  header,\n  data,\n  selection\n\n  /**\n   * Gets / sets the keys to be displayed in the tooltip.\n   * If not set, uses d3.keys(data[key])\n   * @param {string[]} [_=none]\n   * @returns {tooltip | string[]}\n   * @memberof tooltip\n   */\n  tooltip.keys = function(_){return arguments.length ? (keys = _, tooltip) : keys};\n  /**\n   * Gets / sets the values to be displayed next to the keys.\n   * If not set, uses data[key][keys[i]].\n   * If a function, gets passed currentData (data[key]) and keys[i].\n   * @param {*[]} [_=none]\n   * @returns {tooltip | *[]}\n   * @memberof tooltip\n   */\n  tooltip.values = function(_){return arguments.length ? (values = _, tooltip) : values};\n  /**\n   * Gets / sets the header to be displayed in the tooltip.\n   * If not set, uses key\n   * @param {string} [_=none]\n   * @returns {tooltip | string}\n   * @memberof tooltip\n   */\n  tooltip.header = function(_){return arguments.length ? (header = _, tooltip) : header};\n  /**\n   * Gets / sets the data (over the selection) to be used for the tooltip\n   * @param {Object} [_=none]\n   * @returns {tooltip | Object}\n   * @memberof tooltip\n   */\n  tooltip.data = function(_){return arguments.length ? (data = _, tooltip) : data};\n  /**\n   * Gets / sets the selection for the tooltip to be applied on\n   * @param {d3.selection} [_=none]\n   * @returns {tooltip | d3.selection}\n   * @memberof tooltip\n   */\n  tooltip.selection = function(_){return arguments.length ? (selection = _, tooltip) : selection};\n\n  /**\n   * Bind, via selection.on(), the mousemove and mouseout events\n   * @returns undefined\n   */\n  function tooltip( ) {\n    selection.on('mouseover', mousemove)\n    selection.on('mousemove', mousemove)\n    selection.on('mouseout', function(){ d3.selectAll(\".d3sm-tooltip\").remove()})\n  }\n\n\n  /**\n   * Produces the tooltip on mousemove\n   * @param {string} key of the object targeted by the mousemove\n   * @param {number} i (index) of the object targeted by mousemove\n   * @memberof tooltip\n   * @private\n   */\n  function mousemove(key, i) {\n    consoleGroup('d3sm-tooltip')\n    var currentData = data[key]\n\n    var [x, y] = d3.mouse(d3.select(\"html\").node())\n    log('tooltip', 'mousemove detected',{key: key, index: i, x:x, y:y})\n    log('tooltip', 'current data', currentData)\n\n\n\n    var div = safeSelect(d3.select('html'), 'tooltip', 'd3sm-tooltip')\n    .classed('card', true)\n    .style('max-width', '300px')\n    .style('background-color', \"#212529\")\n    .style('color', 'white')\n\n\n\n    var cardBody = safeSelect(div, 'div', 'card-body')\n    var cardTitle = safeSelect(cardBody, 'h5', 'card-title')\n    .text(header == undefined ? key : typeof header == 'function' ? header(key, currentData, i) : header)\n    .style('color', 'cyan')\n\n\n    var table = safeSelect(cardBody, 'table', 'table').classed('table-dark', true)\n    var tBody = safeSelect(table, 'tbody')\n\n    tBody = tBody.selectAll('tr')\n    tBody= tBody.data(keys == undefined ? d3.keys(currentData): keys)\n    tBody.exit().remove()\n\n\n    var tr = tBody.enter().append('tr').style('max-width', '300px')\n    tr.append('td').attr('class', function(d, i){return 'tooltip-key'})\n    tr.append('td').attr('class',  function(d, i, j){return 'tooltip-value'})\n    .attr('tooltip-row-index', function(d, i){return i})\n\n    // tBody = tBody.merge(tr)\n    consoleGroup('tooltip-rows')\n    tBody.selectAll('.tooltip-key').text(function(d, i){return d})\n    tBody.selectAll('tr .tooltip-value')\n    .text(function(d, i){\n      log('tooltip', 'trying to set value', {rowKey: d, rowIndex: i})\n      var i = d3.select(this).attr('tooltip-row-index')\n      var v = currentData[d];\n\n\n      if (values != undefined) {v = values[i]; if(typeof v == \"function\") {v = v(currentData, d)}}\n      return  typeof v == 'number' ? round(v, 5) : v\n    })\n    consoleGroupEnd()\n    consoleGroupEnd()\n\n    x += 15\n    // x += 15\n    var bbox = div.node().getBoundingClientRect()\n    if (x + bbox.width > window.innerWidth - window.scrollX) { x = d3.event.pageX - bbox.width - 15 }\n    if (y + bbox.height > window.innerHeight  - window.scrollY) { y = d3.event.pageY - bbox.height - 15 }\n    div.style('position') == \"relative\"\n    ? div.style('position', 'absolute').style('left', x+'px').style('top', y+'px')\n    : div.style('left', x+'px').style('top', y+'px')\n    // .transition().duration(200).ease(d3.easeSin)\n\n    // if (bbox.x + bbox.width > window.innerWidth) {\n    //   div.style('left', (d3.event.pageX-15-bbox.width)+'px')\n    // }\n    // if (bbox.y + bbox.height > window.innerHeight) {\n    //   div.style('top', (d3.event.pageY-15-bbox.height)+'px')\n    // }\n\n    div.attr('z-index', 10000)\n  }\n\n  return tooltip\n}\n","import {hypenate, safeSelect} from './helpers';\n\nexport function selectFilter(selection) {\n\n  var\n  data,\n  namespace = 'd3sm-select-filter',\n  selectionName = 'Select options:',\n  defaultValue = undefined\n\n\n\n\n  var lastValue = undefined\n\n  selectFilter.data = function(_) { return arguments.length ? (data = _, selectFilter) : data}\n  selectFilter.namespace = function(_) { return arguments.length ? (namespace = _, selectFilter) : namespace}\n  selectFilter.selectionName = function(_) { return arguments.length ? (selectionName = _, selectFilter) : selectionName}\n  selectFilter.defaultValue = function(_) { return arguments.length ? (defaultValue = _, selectFilter) : defaultValue}\n  selectFilter.currentOption = currentOption\n\n  function selectFilter() {\n    var\n    container = safeSelect(selection, 'div', 'input-group').classed(hypenate(namespace,'container'),true),\n\n      selectPrepend = safeSelect(container, 'div', 'select-prepend').classed('input-group-prepend', true),\n        selectPrependSpan = safeSelect(selectPrepend, 'span', 'input-group-text').text(selectionName),\n\n      select = safeSelect(container, 'select', 'custom-select').classed(hypenate(namespace,'select'),true),\n\n      selectAppend = safeSelect(container, 'div', 'select-append').classed('input-group-prepend', true),\n        selectAppendButton = safeSelect(selectAppend, 'a', 'filter-button').classed('btn btn-outline-secondary', true),\n          filterButtonIcon = safeSelect(selectAppendButton, 'i', 'fa fa-filter'),\n\n      inputGroup = safeSelect(container, 'div', 'filter-input-group').classed('input-group',true).classed('d-none', true),\n        inputPrepend = safeSelect(inputGroup, 'div', 'input-group-prepend'),\n          inputPrependSpan = safeSelect(inputPrepend, 'span', 'input-group-text').classed('search-button', true),\n            inputPrependSpanIcon = safeSelect(inputPrependSpan,'i','fa fa-search'),\n\n        input = safeSelect(inputGroup, 'input', 'form-control').attr('placeholder', 'all').attr('type', 'text'),\n        inputAppend = safeSelect(inputGroup, 'div', 'input-group-append'),\n          inputAppendButton = safeSelect(inputAppend, 'a', 'close-button').classed('btn btn-outline-secondary', true),\n            inputAppendButtonIcon = safeSelect(inputAppendButton, 'i', 'fa fa-close')\n\n\n    var keys = d3.keys(data),\n    options = select.selectAll('option')\n\n    options = options.data(d3.keys(data))\n    options = options.merge(options.enter().append('option'))\n    .attr('value', function(d, i){return d})\n    .text(function(d, i){return d})\n\n    var\n    filterButton = selectAppendButton,\n    closeButton = inputAppendButton\n\n    filterButton.on('click', function(d, i){\n      var currentStyle = inputGroup.classed('d-none')\n      inputGroup.classed('d-none', !currentStyle)\n    })\n\n    closeButton.on('click', function(d, i){\n      input.property('value', '').dispatch('input')\n    })\n\n    input.on('input', function(d, i){\n      var\n      val = input.property('value'),\n      reg = new RegExp(val, 'gi'),\n      use\n\n      if (val == '') {use = keys}\n      else {\n        use = []\n        d3.keys(data).map(function(option, j){\n          var match = option.match(reg)\n          if (match == null || match.join('') == '') {}\n          else { use.push(option) }\n        })\n      }\n\n      options = select.selectAll('option')\n      options = options.data(use)\n      options.exit().remove()\n      options = options.merge(options.enter().append('option'))\n      .attr('value', function(d, i){return d})\n      .text(function(d, i){return d})\n\n      var current = currentOption()\n      if (lastValue != current) {\n        lastValue = current\n        select.dispatch('change')\n      }\n    })\n\n\n  }\n\n  function currentOption() {\n    var val = selection.select(\"select\").property('value')\n    return val == undefined || val == ''\n    ? defaultValue == undefined\n      ? d3.keys(data)[0]\n      : defaultValue\n    : val\n  }\n\n  return selectFilter\n}\n","import {hypenate, safeSelect, euclideanDistance} from './helpers';\nimport {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer} from './utils';\nimport {unique, hasQ, flatten, whichBin} from './array-functions';\nimport {groupingSpacer} from './grouping-spacer';\nimport {colorFunction as CF} from './color-function';\nimport {tooltip as TTip} from './tooltip';\nimport './d3-prototypes';\n\nfunction getTranslation(selection){\n  var transform = selection.attr('transform')\n  var [junk, xy] =transform.split('translate(')\n  var [x, y] = xy.split(',')\n  y, junk = y.split(')')\n  return [parseFloat(x), parseFloat(y)]\n}\n\nexport function lasso( selection ) {\n  var\n  svg, // svg that is target of events\n  objectContainer, // container which houses objects we are selecting (allows for transform to be applied to lasso)\n  objectClass, // class of object we are selecting\n  namespace=\"d3sm-lasso\",\n  chartContainer,\n  chartOffset,\n  objectsOffset,\n  eventCatcher,\n\n  xScale, // optional scale for the lasso currentPoints\n  yScale, // optional scale for the lasso currentPoints\n\n  activeQ = false, // whether or not lasso is active\n\n  currentPoints=[], // mouse points for current lasso\n  allPoints=[], // list of lists for all points of lassos\n\n  line = d3.line()\n  .x(function(d, i){\n    var x\n    if (xScale != undefined) { x = xScale(d[0]) }\n    else {x = d[0]}\n    return x //- chartOffset[0]// - objectsOffset[0]\n  })\n  .y(function(d, i){\n    var y\n    if (yScale != undefined) { y= yScale(d[1]) }\n    else {y =  d[1]}\n    return y// - chartOffset[1]// - objectsOffset[1]\n  })\n  .curve(d3.curveLinearClosed),\n\n  instance=0,    // an indentifier for which instance this lasso is under the current svg\n\n  tickDistance = 10,\n\n  // styles for lasso path\n  color = '#17a2b8',\n  animationRate = '10s',\n  opacity=0.3,\n  dashArray = '5, 10',\n  stroke = 'black',\n  strokeWidth=2,\n\n  // styles for lassoed objects\n  lassoedFill = \"white\",\n  lassoedStroke = 'black',\n  lassoedStrokeWidth = 3,\n\n  transitionDuration = 1000,\n  easeFunc = d3.easeExp\n\n  var path\n\n  lasso.svg = function(_) { return arguments.length ? (svg = _, lasso) : svg; }\n  lasso.chartContainer = function(_) { return arguments.length ? (chartContainer = _, lasso) : chartContainer; }\n  lasso.objectContainer = function(_) { return arguments.length ? (objectContainer = _, lasso) : objectContainer; }\n  lasso.objectClass = function(_) { return arguments.length ? (objectClass = _, lasso) : objectClass; }\n  lasso.namespace = function(_) { return arguments.length ? (namespace = _, lasso) : namespace; }\n  lasso.xScale = function(_) { return arguments.length ? (xScale = _, lasso) : xScale; }\n  lasso.yScale = function(_) { return arguments.length ? (yScale = _, lasso) : yScale; }\n  lasso.activeQ = function(_) { return arguments.length ? (activeQ = _, lasso) : activeQ; }\n  lasso.currentPoints = function(_) { return arguments.length ? (currentPoints = _, lasso) : currentPoints; }\n  lasso.allPoints = function(_) { return arguments.length ? (allPoints = _, lasso) : allPoints; }\n  lasso.instance = function(_) { return arguments.length ? (instance = _, lasso) : instance; }\n  lasso.tickDistance = function(_) { return arguments.length ? (tickDistance = _, lasso) : tickDistance; }\n  lasso.color = function(_) { return arguments.length ? (color = _, lasso) : color; }\n  lasso.animationRate = function(_) { return arguments.length ? (animationRate = _, lasso) : animationRate; }\n  lasso.opacity = function(_) { return arguments.length ? (opacity = _, lasso) : opacity; }\n  lasso.dashArray = function(_) { return arguments.length ? (dashArray = _, lasso) : dashArray; }\n  lasso.stroke = function(_) { return arguments.length ? (stroke = _, lasso) : stroke; }\n  lasso.lassoedFill = function(_) { return arguments.length ? (lassoedFill = _, lasso) : lassoedFill; }\n  lasso.lassoedStroke = function(_) { return arguments.length ? (lassoedStroke = _, lasso) : lassoedStroke; }\n  lasso.lassoedStrokeWidth = function(_) { return arguments.length ? (lassoedStrokeWidth = _, lasso) : lassoedStrokeWidth; }\n  lasso.eventCatcher = function(_) { return arguments.length ? (eventCatcher = _, lasso) : eventCatcher; }\n\n  lasso.drag = drag\n  lasso.draw = draw\n  lasso.tick = tick\n  lasso.detect = detect\n  lasso.toggle = toggle\n  lasso.remove = remove\n  lasso.render = render\n  lasso.keyFrames = keyFrames\n  lasso.updateObjects = updateObjects\n  lasso.applyPathAttributes = applyPathAttributes\n  lasso.applyObjectAttributes = applyObjectAttributes\n\n  keyFrames()\n\n  function lasso() {\n    // add a dash animation if needed\n    if (activeQ) { transitionDraw() }\n  }\n\n  function toggle(state) {\n    // use optional param to set state, otherwise toggle state\n    activeQ = (state!=undefined) ? state : !activeQ\n    chartOffset = getTranslation(chartContainer)\n    objectsOffset = getTranslation(objectContainer)\n\n    if (activeQ) {\n      svg.node().addEventListener('mousedown', render, true)\n    } else {\n      svg.node().removeEventListener('mousedown', render, true)\n      remove()\n    }\n\n  }\n\n  function draw() {\n    chartOffset = getTranslation(chartContainer)\n    objectsOffset = getTranslation(objectContainer)\n\n    var container = safeSelect(objectContainer, 'g', 'lasso-container')\n    var paths = container.selectAll('path[instance=\"'+instance+'\"]')\n\n    // update\n    paths = paths.data(allPoints)\n\n    // remove excess\n    var pExit = paths.exit().remove()\n    // add needed paths\n    var pEnter = paths.enter().append('path')\n\n    // merge\n    paths = paths.merge(pEnter)\n\n    // apply\n    applyPathAttributes(paths)\n  }\n\n  function remove() {\n    var container = safeSelect(objectContainer, 'g', 'lasso-container')\n    var paths = container.selectAll('path[instance=\"'+instance+'\"]').remove()\n    container.remove()\n    objectContainer.selectAll(objectClass).classed(\"in-lasso\", false)\n    updateObjects()\n  }\n\n  function render( event ) {\n    // nothing can interefer with drawing the lasso\n    event.preventDefault(); event.stopPropagation();\n\n    var container = safeSelect(objectContainer, 'g', 'lasso-container')\n\n    /*\n    each time the user presses down, while the state is active, the lasso\n    the lasso should make a seperate segment.\n    */\n    currentPoints = [];\n\n    svg.node().addEventListener('mousemove', drag)\n    svg.node().addEventListener('mouseup', function(event) {\n      svg.node().removeEventListener('mousemove', drag)\n      allPoints.push(currentPoints)\n      // BUG:  somehow this is pushing currentPoints n times where n is the nth lasso path for the current instance\n      // NOTE: allPoints = unique(allPoints) is a temporary and inefficient fix\n      allPoints = unique(allPoints)\n    })\n\n    path = container.append('path').data([currentPoints])\n    applyPathAttributes(path)\n  }\n\n  function transitionDraw() {\n    var container = safeSelect(objectContainer, 'g', 'lasso-container')\n    var paths = container.selectAll('path[instance=\"'+instance+'\"]')\n\n    // update\n    paths = paths.data(allPoints)\n\n    // remove excess\n    var pExit = paths.exit().remove()\n    // add needed paths\n    var pEnter = paths.enter().append('path')\n\n    // merge\n    paths = paths.merge(pEnter)\n    .transition().duration(transitionDuration)\n    .ease(easeFunc)\n    applyPathAttributes(paths)\n\n  }\n\n  function applyPathAttributes(path) {\n    path\n    .attr(\"class\", hypenate(namespace, \"lasso-path\"))\n    .style('opacity', opacity)\n    .attr('fill', color)\n    .attr(\"d\", line)\n    .attr('instance', instance)\n    .style(\"stroke-dasharray\", dashArray)\n    .attr(\"stroke\", stroke)\n    .attr(\"stroke-width\", strokeWidth)\n    .style('animation', 'lassoDash '+animationRate+' linear')\n    .style(\"animation-iteration-count\", \"infinite\")\n  }\n\n  function drag(event) {\n    /*\n    effectively create a mouse down and move event (which normally is inteperated\n    as 'drag' by the browser) by dynamically adding / removing this event on\n    mouse down / mouse up.\n    */\n\n    if (eventCatcher != undefined) {eventCatcher.dispatch(hypenate(namespace,\"drag\"))}\n    // d3.dispatch(hypenate(namespace,\"drag\"))\n\n    if (event.which != 1) {return} // ensures left mouse button set\n    d3.event = event\n    var pt = d3.mouse(objectContainer.node());\n    var pt = d3.mouse(svg.node());\n\n    if (xScale != undefined) {pt[0] = xScale.invert(pt[0])}\n    if (yScale != undefined) {pt[1] = yScale.invert(pt[1])}\n    pt[0] = pt[0] - chartOffset[0] - objectsOffset[0]\n    pt[1] = pt[1] - chartOffset[1] - objectsOffset[1]\n\n    /* if we have a point already, test if it passes a minimum distance to prevent overwhelming with too many tick functions */\n    if (currentPoints.length) {\n      var lastPt = currentPoints[currentPoints.length - 1]\n      var a = [pt[0], pt[1]], b = [lastPt[0], lastPt[1]]\n\n      if (xScale) {b[0] = xScale(b[0]); a[0] = xScale(a[0])}\n      if (yScale) {b[1] = yScale(b[1]); a[1] = yScale(a[1])}\n\n      var dist = euclideanDistance(b, a)\n      if (dist > tickDistance) { tick(pt) }\n    }\n    else { tick(pt) }\n  }\n\n\n  function tick (pt) {\n    /*\n    If a point is provided update data and objects.\n    Otherwise just call on data we already have.\n\n    Why like this?:\n    1. currentPoints is current points to allow disjunct lassos, currentPoints is only pushed to\n    allPoints after mouseup.\n    2. to allow render of objects in the lasso class / updating the data list\n    just by toggling the button\n    */\n\n    if (pt != undefined) {\n      currentPoints.push(pt);\n      path.attr(\"d\", line);\n      if (currentPoints.length < 3) {return} // need at least 3 points to detect anything.\n      detect(allPoints.concat([currentPoints]))\n    } else {\n      detect(allPoints)\n    }\n  }\n\n\n  function detect(lassos) {\n    if (lassos == undefined) {lassos = allPoints}\n    objectContainer.selectAll(objectClass).each(function(d, i){\n      var current = d3.select(this),\n\n      box = current.absolutePosition(),\n      // box = current.relativePositionTo(objectContainer.node()),\n\n      boxPts = [\n        [\n          box.left - chartOffset[0] - objectsOffset[0],\n          box.top - chartOffset[1] - objectsOffset[1]\n        ],\n        [\n          box.right - chartOffset[0] - objectsOffset[0],\n          box.top - chartOffset[1] - objectsOffset[1]\n        ],\n        [\n          box.left - chartOffset[0] - objectsOffset[0],\n          box.bottom - chartOffset[1] - objectsOffset[1]\n        ],\n        [\n          box.right - chartOffset[0] - objectsOffset[0],\n          box.bottom - chartOffset[1] - objectsOffset[1]\n        ]\n      ]\n\n      if (xScale != undefined) {\n        boxPts[0][0] = xScale.invert(boxPts[0][0])\n        boxPts[1][0] = xScale.invert(boxPts[1][0])\n        boxPts[2][0] = xScale.invert(boxPts[2][0])\n        boxPts[3][0] = xScale.invert(boxPts[3][0])\n      }\n      if (yScale != undefined) {\n        boxPts[0][1] = yScale.invert(boxPts[0][1])\n        boxPts[1][1] = yScale.invert(boxPts[1][1])\n        boxPts[2][1] = yScale.invert(boxPts[2][1])\n        boxPts[3][1] = yScale.invert(boxPts[3][1])\n      }\n\n\n      /*\n      flag needed as we have to test multiple lasso segments, and if the point\n      is not in one segment, it does not mean it is not in any\n      */\n      var inAnyLassoQ = false;\n      for (var i = 0; i < lassos.length; i++) {\n        var lassoPoints = lassos[i]\n        // .map(function(pt){\n        //   var x, y = pt\n        //   if (xScale!=undefined) {x = xScale(x)}\n        //   if (yScale!=undefined) {y = yScale(y)}\n        //   return [x, y]\n        // })\n        var boxInLassoQ = boxPts.every(coord => d3.polygonContains(lassoPoints, coord))\n\n        if (boxInLassoQ) { inAnyLassoQ = true; } // only update flag in the positive case.\n      }\n\n      current.classed('in-lasso', inAnyLassoQ)\n      current.classed('in-lasso-'+instance, inAnyLassoQ)\n    })\n\n    updateObjects()\n    return objectContainer.selectAll('.in-lasso-'+instance)\n  }\n\n\n\n  function updateObjects() {\n    objectContainer.selectAll(objectClass).each(function(d, i) {\n      var t = d3.select(this)\n      applyObjectAttributes(t, t.classed('in-lasso'))\n    })\n  }\n\n  function applyObjectAttributes(obj, setQ) {\n    var\n    preLassoFill = obj.attr('_pre_lasso_fill'),\n    preLassoStroke = obj.attr('_pre_lasso_stroke'),\n    preLassoStrokeWidth = obj.attr('_pre_lasso_stroke-width')\n\n    if (setQ) {\n      obj.classed(\"in-lasso\", true)\n      obj.classed('in-lasso-'+instance, true)\n      if (preLassoFill == undefined) { obj.attr('_pre_lasso_fill', obj.attr('fill')) }\n      if (preLassoStroke == undefined) { obj.attr('_pre_lasso_stroke', obj.attr('stroke')) }\n      if (preLassoStrokeWidth == undefined) { obj.attr('_pre_lasso_stroke-width', obj.attr('stroke-width')) }\n\n      obj\n      //BUG: when .raise()\n      .attr('fill', lassoedFill)\n      .attr('stroke', lassoedStroke)\n      .attr('stoke-width', lassoedStrokeWidth)\n\n    } else {\n      obj.classed(\"in-lasso\", false)\n      obj.classed('in-lasso-'+instance, false)\n      if (preLassoFill != undefined) { obj.attr('fill', preLassoFill) }\n      if (preLassoStroke != undefined) { obj.attr('stroke', preLassoStroke) }\n      if (preLassoStrokeWidth != undefined) { obj.attr('stroke-width', preLassoStrokeWidth) }\n    }\n  }\n\n  function keyFrames() {\n    var style =\n    d3.select(\"html\").select('style.'+hypenate(namespace,\"lasso-dash\"))\n    if (style.empty()) {\n      d3.select(\"html\").append('style')\n      .classed(hypenate(namespace,\"lasso-dash\"), true)\n      .html(\"@keyframes lassoDash {to { stroke-dashoffset: 1000;}}\")\n    }\n\n  }\n  return lasso\n}\n","import {getContainingSVG} from \"./helpers\";\n/*******************************************************************************\n**                                                                            **\n**                                                                            **\n**                             D3 EXTENSIONS                                  **\n**                                                                            **\n**                                                                            **\n*******************************************************************************/\n/**\n* Recursively ascends parents of selection until it finds an svg tag\n* @function d3.selection.thisSVG\n* @augments d3.selection\n* @returns {Element} which is the svg tag, not the d3 selection of that tag\n*/\nd3.selection.prototype.thisSVG = function() { return getContainingSVG(this.node()); }\n\n\n/**\n* Helper for getting absolute position of the mouse\n* @function d3.mouse.absolute\n* @augments d3.mouse\n* @returns {number[]} [x, y] as they relate to `html` not to local scope.\n*/\nd3.mouse.absolute = function() {\n  var html = d3.select('html').node()\n  var [x, y] = this(html)\n  return [x, y]\n}\n\n\n/**\n* Gets position of the selection in relation to the containing svg\n* @see{@link getContainingSVG}\n* @function d3.selection.absolutePosition\n* @augments d3.selection\n* @returns {Object} with structure similar to getBoundingClientRect, e.g.\n* top, left, bottom, right, height, width\n*/\nd3.selection.prototype.absolutePosition = function() {\n    var element = this.node();\n    var elementPosition = element.getBoundingClientRect();\n    var containerSVG = getContainingSVG(element)\n    var svgPosition = containerSVG.getBoundingClientRect();\n\n    return {\n        top:    elementPosition.top    - svgPosition.top,\n        left:   elementPosition.left   - svgPosition.left,\n        bottom: elementPosition.bottom - svgPosition.top,\n        right:  elementPosition.right  - svgPosition.left,\n        height: elementPosition.height,\n        width:  elementPosition.width\n    };\n\n}\n\n\nd3.selection.prototype.relativePositionTo = function(container) {\n    var element = this.node();\n    var elementPosition = element.getBoundingClientRect();\n    var containerSVG = container\n    var svgPosition = containerSVG.getBoundingClientRect();\n\n    return {\n        top:    elementPosition.top    - svgPosition.top,\n        left:   elementPosition.left   - svgPosition.left,\n        bottom: elementPosition.bottom - svgPosition.top,\n        right:  elementPosition.right  - svgPosition.left,\n        height: elementPosition.height,\n        width:  elementPosition.width\n    };\n\n}\n","// Import styles (automatically inject into <head>).\n// import '../styles/main.css';\nimport {axis} from './modules/axis';\nimport {bar} from './modules/bar';\nimport {bubbleHeatmap} from './modules/bubble-heatmap';\nimport {heatmap} from './modules/heatmap';\nimport {boxwhisker} from './modules/box-whisker';\nimport {colorFunction} from './modules/color-function';\nimport {datatoggle} from './modules/data-toggle';\nimport {groupingSpacer} from './modules/grouping-spacer';\nimport {tooltip} from './modules/tooltip';\nimport {scatter} from './modules/scatter';\nimport {plotZoom} from './modules/plot-zoom';\nimport {multiPlotZoom} from './modules/multi-plot-zoom';\nimport {violin} from './modules/violin';\nimport {numericLegend} from './modules/numeric-legend';\nimport {categoricLegend} from './modules/categorical-legend';\nimport {lasso} from './modules/lasso';\nimport {lassoWidget} from './modules/lasso-widget';\nimport {selectFilter} from './modules/select-filter';\nimport {upset} from './modules/upset';\nimport {filterTable} from './modules/filter-table';\n\nimport {uniqueElements, getTranslation, modifyHexidecimalColorLuminance, tickRange,\nquartiles, extractViolinValues, hypenate, round, getContainingSVG,\ninterpolateColors, truncateText, safeSelect} from './modules/helpers';\n\nimport {\n  all, tally, hasQ, first, last, total, unique, get, listOfListsQ,\n  cut, groupBy, arrayEquals, elementsAtLevels, numberOfElements,\n  flatten, whichBin\n} from './modules/array-functions';\n\n\nimport {\n  setupStandardChartContainers, log as myLog, warn, info, error,\n  consoleGroup, consoleGroupEnd, resizeDebounce\n} from './modules/utils';\n\n// /** @module d3sm */\nvar d3sm = {};\nd3sm.axis = axis;\nd3sm.bar = bar;\nd3sm.bubbleHeatmap = bubbleHeatmap;\nd3sm.heatmap = heatmap;\nd3sm.boxwhisker = boxwhisker;\nd3sm.colorFunction = colorFunction;\nd3sm.datatoggle = datatoggle;\nd3sm.groupingSpacer = groupingSpacer;\nd3sm.tooltip = tooltip;\nd3sm.scatter = scatter;\nd3sm.plotZoom = plotZoom;\nd3sm.multiPlotZoom = multiPlotZoom;\nd3sm.violin = violin;\nd3sm.numericLegend = numericLegend;\nd3sm.categoricLegend = categoricLegend;\nd3sm.lasso = lasso;\nd3sm.lassoWidget = lassoWidget;\nd3sm.selectFilter = selectFilter;\nd3sm.upset = upset;\nd3sm.filterTable = filterTable;\n\nd3sm.uniqueElements = uniqueElements;\nd3sm.getTranslation = getTranslation;\nd3sm.modifyHexidecimalColorLuminance = modifyHexidecimalColorLuminance;\nd3sm.tickRange = tickRange;\nd3sm.quartiles = quartiles;\nd3sm.extractViolinValues = extractViolinValues;\nd3sm.hypenate = hypenate;\nd3sm.round = round;\nd3sm.getContainingSVG = getContainingSVG;\nd3sm.interpolateColors = interpolateColors;\nd3sm.truncateText = truncateText;\nd3sm.safeSelect = safeSelect;\n\nd3sm.whichBin = whichBin;\nd3sm.unique = unique;\nd3sm.flatten = flatten;\n\nd3sm.setupStandardChartContainers = setupStandardChartContainers;\nd3sm.log = myLog;\nd3sm.warn = warn;\nd3sm.info = info;\nd3sm.error = error;\nd3sm.consoleGroup = consoleGroup;\nd3sm.consoleGroupEnd = consoleGroupEnd;\nd3sm.resizeDebounce = resizeDebounce;\n\nd3sm.debugQ = false\n\n\n\n// Import a logger for easier debugging\n// import debug from 'debug';\n// const log = debug('app:log');\n\n// The logger should only be disabled if we're not in production.\n// if (ENV !== 'production') {\n//   // Enable the logger.\n//   debug.enable('*');\n//   log('Logging is enabled!');\n//\n//   // Enable LiveReload\n//   document.write(\n//     '<script src=\"http://'\n//     + (location.host || 'localhost').split(':')[0]\n//     + ':35729/livereload.js?snipver=1\"></'\n//     + 'script>'\n//   );\n// } else {\n//   debug.disable();\n// }\n\nwindow.d3sm = d3sm;\n\nexport default d3sm\n","import {\n  hypenate, safeSelect, extractViolinValues,\n  tickRange, modifyHexidecimalColorLuminance, truncateText,\n  truncateString,\n  round\n} from './helpers';\nimport {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer} from './utils';\nimport {unique, hasQ, flatten} from './array-functions';\nimport {groupingSpacer} from './grouping-spacer';\n/*******************************************************************************\n**                                                                            **\n**                                                                            **\n**                                  AXIS                                      **\n**                                                                            **\n**                                                                            **\n*******************************************************************************/\n\n/**\n * Creates an axis\n *\n * {@link https://sumneuron.gitlab.io/d3sm/demos/axes/index.html Demo}\n * @constructor axis\n * @param {d3.selection} selection\n * @namespace axis\n * @returns {function} axis\n */\nexport function axis ( selection ) {\n  var\n  /**\n  * The orientation of the axis\n  * (see {@link axis#orient})\n  * @param {string} [orient='bottom']\n  * @memberof axis#\n  * @property\n  */\n  orient = 'bottom',       // direction of the axis\n\n  /**\n  * Amount of horizontal space (in pixels) avaible to render the axis in\n  * (see {@link axis#spaceX})\n  * @param {number} [spaceX=0]\n  * @memberof axis#\n  * @property\n  */\n  spaceX=0,\n  /**\n  * Amount of vertical space (in pixels) avaible to render the axis in\n  * (see {@link axis.spaceY})\n  * @param {number} [spaceY=0]\n  * @memberof axis#\n  * @property\n  */\n  spaceY=0,\n\n\n  /**\n  * Whether or not to allow axis to render elements pass the main spatial dimension\n  * given the orientation (see {@link axis#orient}), where {@link axis#orient}=\"bottom\" or {@link axis#orient}=\"top\"\n  * the main dimension is {@link axis#spaceX} and where {@link axis#orient}=\"left\" or {@link axis#orient}=\"right\"\n  * the main dimension is {@link axis#spaceY}\n  * @param {boolean} [overflowQ=false]\n  * @memberof axis#\n  * @property\n  */\n  overflowQ = false,    // whether or not to allow overflow\n  /**\n  * Whether or not the axis labels are for categorical data. If false,\n  * will use {@link axis#scale} to position ticks.\n  * @param {boolean} [categoricalQ=false]\n  * @memberof axis#\n  * @property\n  */\n  categoricalQ = false, // whether or not the axis is showing values or groups\n  /**\n  * Whether or not the axis ticks should have guidelines\n  * @param {boolean} [categoricalQ=false]\n  * @memberof axis#\n  * @property\n  */\n  guideLinesQ = false,    // whether or not to allow overflow\n\n\n  /**\n  * How to group the tick labels\n  * @param {Array[]} [grouping=undefined] list of putatively other lists, which should correspond to tickLabels\n  * will space tick labels in nested lists closer together than outer lists\n  * @memberof axis#\n  * @property\n  */\n  grouping,\n\n  /**\n  * The scale for which non-categorial (see {@link axis#categoricalQ}) ticks should be spaced\n  * @param {d3.scale} [scale=d3.scaleLinear]\n  * @memberof axis#\n  * @property\n  */\n\n  scale = d3.scaleLinear(),\n  /**\n  * The padding for the domain of the scale (see {@link axis#scale})\n  * @param {d3.scale} [scale=d3.scaleLinear]\n  * @memberof axis#\n  * @property\n  */\n  domainPadding = 0.5,\n\n\n  /**\n  * Default space for the spacer (percentage) of main dimension given the orientation\n  * (see {@link axis#orient}), where {@link axis#orient}=\"bottom\" or {@link axis#orient}=\"top\"\n  * the main dimension is {@link axis#spaceX} and where {@link axis#orient}=\"left\" or {@link axis#orient}=\"right\"\n  * the main dimension is {@link axis#spaceY}between ticks\n  * @param {number} [objectSpacer=0.05]\n  * @memberof axis#\n  * @property\n  */\n  objectSpacer = 0.05,\n  /**\n  * The minimum size that an object can be if {@link axis#categoricalQ} is set to true\n  * @param {number} [minObjectSize=15]\n  * @memberof axis#\n  * @property\n  */\n  minObjectSize = 15,\n  /**\n  * The maximum size that an object can be if {@link axis#categoricalQ} is set to true\n  * @param {number} [maxObjectSize=15]\n  * @memberof axis#\n  * @property\n  */\n  maxObjectSize = 50,\n\n  /**\n  * Color of the background\n  * @param {string} [backgroundFill=\"transparent\"]\n  * @memberof axis#\n  * @property\n  */\n  backgroundFill = 'transparent',\n  /**\n  * Namespace for all items made by this instance of axis\n  * @param {string} [namespace=\"d3sm-axis\"]\n  * @memberof axis#\n  * @property\n  */\n  namespace = 'd3sm-axis',\n  /**\n  * Class name for tick container (<g> element)\n  * @param {string} [objectClass=\"tick-group\"]\n  * @memberof axis#\n  * @property\n  */\n  objectClass = 'tick-group',\n\n  /**\n  * Values to show at each tick. Only used if categoricalQ is set true. See {@link axis#categoricalQ}\n  * @param {string[]} [tickLabels=undefined]\n  * @memberof axis#\n  * @property\n  */\n  tickLabels,   // what to place at ticks\n  /**\n  * Values to show at each tick. Only used if categoricalQ is set false. See {@link axis#categoricalQ}\n  * @param {string[] | number[]} [objectClass=undefined]\n  * @memberof axis#\n  * @property\n  */\n  tickValues,   // where to place ticks if not\n  /**\n  * Number of ticks to display if categoricalQ is false. See {@link axis#categoricalQ}\n  * @param {number} [numberOfTicks=5]\n  * @memberof axis#\n  * @property\n  */\n  numberOfTicks = 5,\n\n\n  /**\n  * Stroke color of the main axis line\n  * @param {string} [lineStroke='black']\n  * @memberof axis#\n  * @property\n  */\n  lineStroke = 'black',\n  /**\n  * Stroke width of the main axis line\n  * @param {number} [lineStrokeWidth=3]\n  * @memberof axis#\n  * @property\n  */\n  lineStrokeWidth = 3,\n\n\n  /**\n  * Stroke color of ticks\n  * @param {string} [tickStroke='black']\n  * @memberof axis#\n  * @property\n  */\n  tickStroke = 'black',\n  /**\n  * Stroke number of ticks\n  * @param {string} [tickStrokeWidth=2]\n  * @memberof axis#\n  * @property\n  */\n  tickStrokeWidth = 2,\n  /**\n  * Length - in pixels - of ticks\n  * @param {number} [tickLength=10]\n  * @memberof axis#\n  * @property\n  */\n  tickLength = 10,\n\n  tickTickLabelSpacer = 10,\n  tickLabelMargin = 10,\n\n\n  /**\n  * Font size of tick labels\n  * @param {number} [tickLabelFontSize=14]\n  * @memberof axis#\n  * @property\n  */\n  tickLabelFontSize = 14,\n  /**\n  * Min font size of tick labels\n  * @param {number} [tickLabelMinFontSize=8]\n  * @memberof axis#\n  * @property\n  */\n  tickLabelMinFontSize = 8,\n  /**\n  * Max font size of tick labels\n  * @param {number} [tickLabelMaxFontSize=20]\n  * @memberof axis#\n  * @property\n  */\n  tickLabelMaxFontSize = 20,\n\n\n  /**\n  * Text anchor of tick labels\n  * @param {string} [tickLabelTextAnchor=\"middle\"]\n  * @memberof axis#\n  * @property\n  */\n  tickLabelTextAnchor,\n  /**\n  * Rotation of tick labels\n  * @param {number} [tickLabelRotation=0]\n  * @memberof axis#\n  * @property\n  */\n  tickLabelRotation,\n  /**\n  * Optional function for extracting the tick label from data\n  * @param {function} [tickLabelFunc=undefined]\n  * @memberof axis#\n  * @property\n  */\n  tickLabelFunc = undefined,\n\n  /**\n  * Optional function for what to do when label is clicked\n  * @param {function} [tickLabelOnClick=function(d, i){}]\n  * @memberof axis#\n  * @property\n  */\n  tickLabelOnClick = function(d, i){},\n\n  /**\n  * Optional function for what to do when label is hovered\n  * @param {function} [tickLabelOnHoverFunc=function(d, i){}]\n  * @memberof axis#\n  * @property\n  */\n  tickLabelOnHoverFunc = function(d, i){\n    return String(d).replace('-', ' ').replace('_', ' ')\n  },\n\n\n  /**\n  * Length of guidelines\n  * @param {function} [guidelineSpace=undefined]\n  * @memberof axis#\n  * @property\n  */\n  guidelineSpace,\n  /**\n  * Stroke color of guidlines\n  * @param {string} [guidelineSpace=\"#333333\"]\n  * @memberof axis#\n  * @property\n  */\n  guideLineStroke = '#333333',\n  /**\n  * Stroke width of guidlines\n  * @param {number} [guidelineSpace=2]\n  * @memberof axis#\n  * @property\n  */\n  guideLineStrokeWidth = 2,\n\n  /**\n  * Duration of all transitions of this element\n  * @param {number} [transitionDuration=1000]\n  * @memberof axis#\n  * @property\n  */\n  transitionDuration = 1000,\n  /**\n  * Easing function for transitions\n  * @param {d3.ease} [easeFunc=d3.easeExp]\n  * @memberof axis#\n  * @property\n  */\n  easeFunc = d3.easeExp,\n\n\n  /**\n  * Closure variable for getting object size after calculation\n  * @param {number} [objectSize=undefined]\n  * @memberof axis#\n  * @property\n  */\n  objectSize,\n  /**\n  * Closure variable for getting spacer size after calculation\n  * @param {number} [spacerSize=undefined]\n  * @memberof axis#\n  * @property\n  */\n  spacerSize,\n\n  /**\n  * Decimal percision to round numerical tick labels to\n  * @param {number} [roundTo=2]\n  * @memberof axis#\n  * @property\n  */\n  roundTo = 2,\n\n  label,\n\n\n  reverseScaleQ = false\n\n  axis.label = function(_) { return arguments.length ? (label = _, axis) : label; };\n  axis.tickTickLabelSpacer = function(_) { return arguments.length ? (tickTickLabelSpacer = _, axis) : tickTickLabelSpacer; };\n  axis.tickLabelMargin = function(_) { return arguments.length ? (tickLabelMargin = _, axis) : tickLabelMargin; };\n  /**\n   * Gets or sets the selection in which items are manipulated\n   * @param {d3.selection} [_=none]\n   * @returns {axis | d3.selection}\n   * @memberof axis\n   * @property\n   * by default selection = selection\n   */\n\n  axis.selection = function(_) { return arguments.length ? (selection = _, axis) : selection; };\n\n  /**\n   * Gets or sets the orientation in which items are manipulated\n   * (see {@link axis#orient})\n   * @param {string} [_=none] should be horizontal or vertical\n   * @returns {axis | string}\n   * @memberof axis\n   * @property\n   * by default orient=\"bottom\"\n   */\n  axis.orient = function(_) { return arguments.length ? (orient = _, axis) : orient; };\n  /**\n   * Gets or sets the amount of horizontal space in which items are manipulated\n   * (see {@link axis#spaceX})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default spaceX = undefined\n   */\n  axis.spaceX = function(_) { return arguments.length ? (spaceX = _, axis) : spaceX; };\n  /**\n   * Gets or sets the amount of vertical space in which items are manipulated\n   * (see {@link axis#spaceY})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default spaceY = undefined\n   */\n  axis.spaceY = function(_) { return arguments.length ? (spaceY = _, axis) : spaceY; };\n\n\n  /**\n   * Gets / sets whether or not axis is allowed to go beyond specified dimensions\n   * (see {@link axis#spaceX})\n   * @param {boolean} [_=none]\n   * @returns {axis | boolean}\n   * @memberof axis\n   * @property\n   * by default overflowQ = false\n   */\n  axis.overflowQ = function(_) { return arguments.length ? (overflowQ = _, axis) : overflowQ; };\n  /**\n   * Gets / sets whether or not axis will display categorial ticks or by numerical value\n   * (see {@link axis#categoricalQ})\n   * @param {boolean} [_=none]\n   * @returns {axis | boolean}\n   * @memberof axis\n   * @property\n   * by default categoricalQ = false\n   */\n  axis.categoricalQ = function(_) { return arguments.length ? (categoricalQ = _, axis) : categoricalQ; };\n  /**\n   * Gets / sets whether or not axis ticks should have guidelines\n   * (see {@link axis#guideLinesQ})\n   * @param {boolean} [_=none]\n   * @returns {axis | boolean}\n   * @memberof axis\n   * @property\n   * by default guideLinesQ = false\n   */\n  axis.guideLinesQ = function(_) { return arguments.length ? (guideLinesQ = _, axis) : guideLinesQ; };\n\n\n  /**\n   * Gets / sets how ticks should be groupped\n   * (see {@link axis#grouping})\n   * @param {Array[]} [_=none] list of putatively other lists, which should correspond to tickLabels\n   * will space tick labels in nested lists closer together than outer lists\n   * @returns {axis | Array[]}\n   * @memberof axis\n   * @property\n   * by default grouping = undefined\n   */\n  axis.grouping = function(_) { return arguments.length ? (grouping = _, axis) : grouping; };\n\n\n  /**\n   * Gets / sets the scale for which non-categorial  ticks should\n   * be spaced\n   * (see {@link axis#scale})\n   * @param {d3.scale} [_=none]\n   * @returns {axis | d3.scale}\n   * @memberof axis\n   * @property\n   * by default scale = d3.scaleLinear()\n   */\n  axis.scale = function(_) { return arguments.length ? (scale = _, axis) : scale; };\n  /**\n   * Gets / sets the padding for the domain of the scale\n   * (see {@link axis#domainPadding})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default domainPadding = 0.5\n   */\n  axis.domainPadding = function(_) { return arguments.length ? (domainPadding = _, axis) : domainPadding; };\n\n\n  /**\n   * Gets / sets objectSpacer\n   * (see {@link axis#objectSpacer})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default objectSpacer = 0.05\n   */\n  axis.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, axis) : objectSpacer; };\n  /**\n   * Gets / sets the minObjectSize\n   * (see {@link axis#minObjectSize})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default minObjectSize = 15\n   */\n  axis.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, axis) : minObjectSize; };\n  /**\n   * Gets / sets the maxObjectSize\n   * (see {@link axis#maxObjectSize})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default maxObjectSize = 50\n   */\n  axis.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, axis) : maxObjectSize; };\n\n\n  /**\n   * Gets / sets the namespace\n   * (see {@link axis#namespace})\n   * @param {string} [_=none]\n   * @returns {axis | string}\n   * @memberof axis\n   * @property\n   * by default namespace = 'd3sm-axis'\n   */\n  axis.namespace = function(_) { return arguments.length ? (namespace = _, axis) : namespace; };\n  /**\n   * Gets / sets the backgroundFill\n   * (see {@link axis#backgroundFill})\n   * @param {string} [_=none]\n   * @returns {axis | string}\n   * @memberof axis\n   * @property\n   * by default backgroundFill = 'transparent'\n   */\n  axis.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, axis) : backgroundFill; };\n  /**\n   * Gets / sets the objectClass\n   * (see {@link axis#objectClass})\n   * @param {string} [_=none]\n   * @returns {axis | string}\n   * @memberof axis\n   * @property\n   * by default objectClass = 'tick-group'\n   */\n  axis.objectClass = function(_) { return arguments.length ? (objectClass = _, axis) : objectClass; };\n\n\n  /**\n   * Gets / sets the tickLabels\n   * (see {@link axis#tickLabels})\n   * @param {string[]} [_=none]\n   * @returns {axis | string[]}\n   * @memberof axis\n   * @property\n   * by default tickLabels = undefined\n   */\n  axis.tickLabels = function(_) { return arguments.length ? (tickLabels = _, axis) : tickLabels; };\n  /**\n   * Gets / sets the tickValues\n   * (see {@link axis#tickValues})\n   * @param {number[]} [_=none]\n   * @returns {axis | number[]}\n   * @memberof axis\n   * @property\n   * by default tickValues = undefined\n   */\n  axis.tickValues = function(_) { return arguments.length ? (tickValues = _, axis) : tickValues; };\n  /**\n   * Gets / sets the tickValues\n   * (see {@link axis#numberOfTicks})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default numberOfTicks = 5\n   */\n  axis.numberOfTicks = function(_) { return arguments.length ? (numberOfTicks = _, axis) : numberOfTicks; };\n\n\n  /**\n   * Gets / sets the lineStroke\n   * (see {@link axis#lineStroke})\n   * @param {string} [_=none]\n   * @returns {axis | string}\n   * @memberof axis\n   * @property\n   * by default lineStroke = 'black'\n   */\n  axis.lineStroke = function(_) { return arguments.length ? (lineStroke = _, axis) : lineStroke; };\n  /**\n   * Gets / sets the lineStrokeWidth\n   * (see {@link axis#lineStrokeWidth})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default lineStrokeWidth = 3\n   */\n  axis.lineStrokeWidth = function(_) { return arguments.length ? (lineStrokeWidth = _, axis) : lineStrokeWidth; };\n\n\n  /**\n   * Gets / sets the tickStroke\n   * (see {@link axis#tickStroke})\n   * @param {string} [_=none]\n   * @returns {axis | string}\n   * @memberof axis\n   * @property\n   * by default tickStroke = 'black'\n   */\n  axis.tickStroke = function(_) { return arguments.length ? (tickStroke = _, axis) : tickStroke; };\n  /**\n   * Gets / sets the tickStrokeWidth\n   * (see {@link axis#tickStrokeWidth})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default tickStrokeWidth = 2\n   */\n  axis.tickStrokeWidth = function(_) { return arguments.length ? (tickStrokeWidth = _, axis) : tickStrokeWidth; };\n  /**\n   * Gets / sets the tickLength\n   * (see {@link axis#tickLength})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default tickLength = 10\n   */\n  axis.tickLength = function(_) { return arguments.length ? (tickLength = _, axis) : tickLength; };\n\n\n  /**\n   * Gets / sets the tickLabelFontSize\n   * (see {@link axis#tickLabelFontSize})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default tickLabelFontSize = 14\n   */\n  axis.tickLabelFontSize = function(_) { return arguments.length ? (tickLabelFontSize = _, axis) : tickLabelFontSize; };\n  /**\n   * Gets / sets the tickLabelMinFontSize\n   * (see {@link axis#tickLabelMinFontSize})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default tickLabelMinFontSize = 8\n   */\n  axis.tickLabelMinFontSize = function(_) { return arguments.length ? (tickLabelMinFontSize = _, axis) : tickLabelMinFontSize; };\n  /**\n   * Gets / sets the tickLabelMaxFontSize\n   * (see {@link axis#tickLabelMaxFontSize})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default tickLabelMaxFontSize = 20\n   */\n  axis.tickLabelMaxFontSize = function(_) { return arguments.length ? (tickLabelMaxFontSize = _, axis) : tickLabelMaxFontSize;};\n\n\n  /**\n   * Gets / sets the tickLabelTextAnchor\n   * (see {@link axis#tickLabelTextAnchor})\n   * @param {string} [_=none]\n   * @returns {axis | string}\n   * @memberof axis\n   * @property\n   * by default tickLabelTextAnchor = 'center'\n   */\n  axis.tickLabelTextAnchor = function(_) { return arguments.length ? (tickLabelTextAnchor = _, axis) : tickLabelTextAnchor; };\n  /**\n   * Gets / sets the tickLabelRotation\n   * (see {@link axis#tickLabelRotation})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default tickLabelRotation = 0\n   */\n  axis.tickLabelRotation = function(_) { return arguments.length ? (tickLabelRotation = _, axis) : tickLabelRotation; };\n  /**\n   * Gets / sets the tickLabelFunc\n   * (see {@link axis#tickLabelFunc})\n   * @param {function} [_=none]\n   * @returns {axis | function}\n   * @memberof axis\n   * @property\n   * by default tickLabelFunc = undefined\n   */\n  axis.tickLabelFunc = function(_) { return arguments.length ? (tickLabelFunc = _, axis) : tickLabelFunc; };\n\n\n  /**\n  * Gets / sets the tickLabelOnClick\n  * (see {@link axis#tickLabelOnClick})\n  * @param {function} [_=none]\n  * @returns {axis | function}\n  * @memberof axis\n  * @property\n  */\n  axis.tickLabelOnClick = function(_) { return arguments.length ? (tickLabelOnClick = _, axis) : tickLabelOnClick; };\n\n\n  /**\n   * Gets / sets the guidelineSpace\n   * (see {@link axis#guidelineSpace})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default guidelineSpace = undefined\n   */\n  axis.guidelineSpace = function(_) { return arguments.length ? (guidelineSpace = _, axis) : guidelineSpace; };\n  /**\n   * Gets / sets the guideLineStroke\n   * (see {@link axis#guideLineStroke})\n   * @param {string} [_=none]\n   * @returns {axis | string}\n   * @memberof axis\n   * @property\n   * by default guideLineStroke = \"#333333\"\n   */\n  axis.guideLineStroke = function(_) { return arguments.length ? (guideLineStroke = _, axis) : guideLineStroke; };\n  /**\n   * Gets / sets the guideLineStrokeWidth\n   * (see {@link axis#guideLineStrokeWidth})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default guideLineStrokeWidth = 2\n   */\n  axis.guideLineStrokeWidth = function(_) { return arguments.length ? (guideLineStrokeWidth = _, axis) : guideLineStrokeWidth; };\n\n\n  /**\n   * Gets / sets the transitionDuration\n   * (see {@link axis#transitionDuration})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default transitionDuration = 1000\n   */\n  axis.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, axis) : transitionDuration; };\n  /**\n   * Gets / sets the easeFunc\n   * (see {@link axis#easeFunc})\n   * @param {d3.ease} [_=none]\n   * @returns {axis | d3.ease}\n   * @memberof axis\n   * @property\n   * by default easeFunc = d3.easeExp\n   */\n  axis.easeFunc = function(_) { return arguments.length ? (easeFunc = _, axis) : easeFunc; };\n\n\n  /**\n   * Gets / sets the objectSize\n   * (see {@link axis#objectSize})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default objectSize = undefined\n   */\n  axis.objectSize = function(_) { return arguments.length ? (objectSize = _, axis) : objectSize; };\n  /**\n   * Gets / sets the spacerSize\n   * (see {@link axis#spacerSize})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default spacerSize = undefined\n   */\n  axis.spacerSize = function(_) { return arguments.length ? (spacerSize = _, axis) : spacerSize; };\n\n  /**\n   * Gets / sets the roundTo\n   * (see {@link axis#roundTo})\n   * @param {number} [_=none]\n   * @returns {axis | number}\n   * @memberof axis\n   * @property\n   * by default roundTo = 2\n   */\n   axis.roundTo = function(_) { return arguments.length ? (roundTo = _, axis) : roundTo; };\n   axis.reverseScaleQ = function(_) { return arguments.length ? (reverseScaleQ = _, axis) : reverseScaleQ; };\n\n\n   axis.tickLabelOnHoverFunc = function(_) {return arguments.length ? (tickLabelOnHoverFunc = _, axis) : tickLabelOnHoverFunc; };\n\n\n  function axis () {\n    // for convenience in handling orientation specific values\n    var horizontalQ = hasQ(['top', 'bottom', 'horizontal'], orient) ? true : false\n    var verticalQ = !horizontalQ\n\n    // background cliping rectangle\n    var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}\n    // modify the rect based on axis orientation\n    if (orient == \"left\") {\n      bgcpRect.x -= spaceX;\n      if(guideLinesQ) { bgcpRect.width += guidelineSpace };\n      /* these two lines increase the clipping rect to allow for text at the edge of the axis */\n      bgcpRect.y -= tickLabelMaxFontSize;\n      bgcpRect.height += 2*tickLabelMaxFontSize\n    }\n    if (orient == \"bottom\"){\n      bgcpRect.y = bgcpRect.y;\n      if(guideLinesQ) { bgcpRect.y -= guidelineSpace; bgcpRect.height += guidelineSpace; };\n      /* these two lines increase the clipping rect to allow for text at the edge of the axis */\n      bgcpRect.x -= tickLabelMaxFontSize;\n      bgcpRect.width += 2*tickLabelMaxFontSize\n    }\n    if (orient == \"top\") {\n      bgcpRect.y -= spaceY;\n      if(guideLinesQ) { bgcpRect.height += guidelineSpace };\n      /* these two lines increase the clipping rect to allow for text at the edge of the axis */\n      bgcpRect.y -= tickLabelMaxFontSize;\n      bgcpRect.height += 2*tickLabelMaxFontSize\n    }\n    if (orient == \"right\") { bgcpRect.x = 0;\n      if(guideLinesQ) { bgcpRect.width += guidelineSpace; bgcpRect.x -= guidelineSpace };\n      /* these two lines increase the clipping rect to allow for text at the edge of the axis */\n      bgcpRect.y -= tickLabelMaxFontSize;\n      bgcpRect.height += 2*tickLabelMaxFontSize\n    }\n\n\n    var container = setupContainer( selection, namespace, bgcpRect, backgroundFill );\n\n    // defaults for text-anchor and text rotation\n    if (orient == 'top') {\n      tickLabelTextAnchor = tickLabelTextAnchor == undefined ? 'start' : tickLabelTextAnchor\n      tickLabelRotation = tickLabelRotation == undefined ? -90 : tickLabelRotation\n    }\n    if (orient == 'bottom') {\n      tickLabelTextAnchor = tickLabelTextAnchor == undefined ? 'end' : tickLabelTextAnchor\n      tickLabelRotation = tickLabelRotation == undefined ? -90 : tickLabelRotation\n    }\n    if (orient == 'left') {\n      tickLabelTextAnchor = tickLabelTextAnchor == undefined ? 'end' : tickLabelTextAnchor\n      tickLabelRotation = tickLabelRotation == undefined ? 0 : tickLabelRotation\n    }\n    if (orient == 'right') {\n      tickLabelTextAnchor = tickLabelTextAnchor == undefined ? 'start' : tickLabelTextAnchor\n      tickLabelRotation = tickLabelRotation == undefined ? 0 : tickLabelRotation\n    }\n\n    /*\n    If categorical:\n      -> use grouping if defined,\n      -> else use the labels provided\n    else:\n      if grouping undefined\n        and no specified number of tickes\n          -> make numberOfTick ticks\n          -> else use provided tick values\n        -> use grouping\n    */\n    var tickData = categoricalQ\n    ? (grouping == undefined)\n      ? tickLabels\n      : grouping\n    : (grouping == undefined)\n      ? (numberOfTicks != undefined)\n      // ? (tickValues.length < numberOfTicks)\n        ? (tickRange(...d3.extent(tickValues), numberOfTicks))\n        : tickValues\n      : grouping\n\n\n    var flatTickData = flatten(tickData)\n    var numberOfObjects = flatTickData.length\n    var space = horizontalQ ? spaceX : spaceY\n    var extent = d3.extent(flatTickData)\n\n\n    if (reverseScaleQ) {extent.reverse()}\n    var domain = reverseScaleQ\n    ? [extent[0] + domainPadding, extent[1] - domainPadding]\n    : [extent[0] - domainPadding, extent[1] + domainPadding]\n\n    scale\n    .domain(domain)\n    .range([horizontalQ ? 0 : spaceY, horizontalQ ? spaceX : 0])\n\n\n    /*\n    Scales are based on the values of the chart and correspond to the spacings of the\n    chart. If the chart has already been rendered, these values (expensive to caluclate) can\n    be passed to axis to prevent recalculation.\n    */\n\n    // calculate object size if needed\n    objectSize = (objectSize == undefined)\n    ? calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ)\n    : objectSize\n\n    // calculate spacer size if needed\n    spacerSize = (spacerSize == undefined)\n    ? calculateWidthOfSpacer(flatTickData, space, objectSize, numberOfObjects, objectSpacer, overflowQ)\n    : spacerSize\n\n\n    var objClass = hypenate(namespace, categoricalQ ? objectClass+'-categorical' : objectClass)\n\n    var spacerFunction = groupingSpacer()\n    .horizontalQ(horizontalQ).scale(scale).moveby((categoricalQ?'category':'scale')).numberOfObjects(numberOfObjects)\n    .objectClass(objClass).objectSize(objectSize).spacerSize(spacerSize)\n    .transitionDuration(transitionDuration).easeFunc(easeFunc)\n    .namespace(namespace)\n\n    var tickEnterAnimation = function(sel){\n      var mt = scale(sel.datum()),\n      dist = scale(extent[1]) * 2,\n      k = (mt < extent[1] / 2) ? 1 : -1\n      k = horizontalQ ? k * -1 : k\n      sel.attr('transform', function (d, i) {\n        var\n        x = horizontalQ ?  dist * k : 0,\n        y = !horizontalQ ? dist * k : 0,\n        t = 'translate('+x+','+y+')'\n        return t\n      })\n    }\n    var tickExitAnimation = function(sel) {\n      var mt = scale(sel.datum()),\n      dist = scale(extent[1]) * 2,\n      k = (mt < extent[1] / 2) ? 1 : -1\n      k = horizontalQ ? k * -1 : k\n      sel.transition().duration(transitionDuration).ease(easeFunc)\n      .style('opacity', 0)\n      .attr('transform', function (d, i) {\n        var\n\n        x = horizontalQ ?  dist * k  : 0,\n        y = !horizontalQ ? dist * k : 0,\n        t = 'translate('+x+','+y+')'\n        return t\n      }).remove()\n    }\n\n    if (!categoricalQ){\n      spacerFunction.enterFunction(tickEnterAnimation)\n      spacerFunction.exitFunction(tickExitAnimation)\n    }\n\n    // move tick containers\n    spacerFunction(container, tickData, 0)\n\n    // move by for x and y needed to center categorical ticks, labels, and guidelines\n    function moveXBy(d, i, horizontalQ, categoricalQ, objectSize){\n      return (horizontalQ)\n      ? (categoricalQ)\n        ? objectSize / 2\n        : 0\n      : 0\n    }\n\n    function moveYBy(d, i, verticalQ, categoricalQ, objectSize){\n      return (verticalQ)\n      ? (categoricalQ)\n        ? objectSize / 2\n        : 0\n      : 0\n    }\n\n\n\n    var labelNameGroup = safeSelect(selection, 'g', hypenate(namespace,'axis-name'))\n\n\n\n    var labelElement = safeSelect(labelNameGroup, 'text', hypenate(namespace, 'name'))\n    if (labelElement != undefined) {\n      labelElement.text(label)\n\n      if (orient == 'left' || orient == 'right') {\n        labelElement.attr('transform', 'rotate(-90)')\n      }\n\n      var bbox = labelElement.node().getBoundingClientRect()\n      labelNameGroup.attr('transform',function(d, i){\n        var\n        x = 0,\n        y = 0,\n        t\n\n        if (orient == 'bottom') {\n          x = spaceX - bbox.width - tickLabelMargin\n          y = - tickTickLabelSpacer\n        }\n        else if (orient == 'top') {\n          x = spaceX - bbox.width - tickLabelMargin\n          x = tickLabelMargin\n          y = bbox.height + tickTickLabelSpacer\n        }\n        else if (orient == 'left') {\n          x = bbox.width + tickTickLabelSpacer\n          y = bbox.height + tickLabelMargin\n        } else if (orient == 'right') {\n          x = -(bbox.width + tickTickLabelSpacer)\n          y = bbox.height + tickLabelMargin\n        } else {\n\n        }\n        t = 'translate('+x+','+y+')'\n        return t\n      })\n\n\n\n    } else {\n      labelElement.remove()\n    }\n    /*\n    Idea from Stack Overflow\n    https://stackoverflow.com/questions/50579535/d3-js-v4-truncate-text-to-fit-in-fixed-space/50585022?noredirect=1#comment88235562_50585022\n    to use clip path to make things fit in fixed size. Have yet got this to work nicely.\n    */\n    // var defs = d3.select(container.node().parentNode).select('defs')\n    // var tickLabelClipPath = safeSelect(defs, 'clipPath', hypenate(namespace,'tick-label-clip-path')).attr('id',  hypenate(namespace,'tick-label-clip-path'))\n    // var tickLabelClipPathRect = safeSelect(tickLabelClipPath, 'rect',  hypenate(namespace,'tick-label-clip-path-rect'))\n    // .attr('x', 0)\n    // .attr('y', 0)\n    // .attr('width', function(d, i){\n    //   if (horizontalQ) { return tickLabelFontSize }\n    //   if (verticalQ) { return spaceX - tickLength }\n    // })\n    // .attr('height', function(d, i){\n    //   if (verticalQ) { return tickLabelFontSize }\n    //   if (horizontalQ) { return spaceY - tickLength }\n    // })\n\n\n    // for each tick container\n    var ticks = container.selectAll('g:not(.to-remove).'+objClass).each(function(d, i){\n      var that = d3.select(this).style('opacity', 1)\n\n      // make and move tick\n      var tick = safeSelect(that, 'line', hypenate(namespace,'tick'))\n      .attr(\"x1\", 0)\n      .attr(\"x2\", horizontalQ ? 0 : orient == \"left\" ? -tickLength : tickLength)\n      .attr(\"y1\", 0)\n      .attr('y2',  verticalQ ? 0 : orient == \"top\" ? -tickLength : tickLength)\n      .attr('stroke', tickStroke)\n      .attr('stroke-width', tickStrokeWidth)\n      .attr('transform', function(d, i) {\n        var\n        x = moveXBy(d, i, horizontalQ, categoricalQ, objectSize),\n        y = moveYBy(d, i, verticalQ, categoricalQ, objectSize),\n        t = 'translate('+x+','+y+')'\n        return t\n      })\n\n      // make and move label\n      var label = safeSelect(that, 'text', hypenate(namespace,'label'))\n      .text(function(d, i){\n        var s = typeof d == 'number' ? round(d, roundTo) : d\n        s = truncateString(String(s), (horizontalQ ? spaceY : spaceX) - tickLength-tickLabelMargin-tickTickLabelSpacer, tickLabelFontSize * 0.45)\n        return s\n      })\n      .attr('font-size', tickLabelFontSize)\n      .attr('text-anchor', tickLabelTextAnchor)\n      // truncateText(label, label.text(), orient, tickLength, horizontalQ ? spaceY : spaceX, overflowQ)\n\n      label.attr('transform', function(d, i) {\n        var\n        rect = d3.select(this).node().getBoundingClientRect(),\n        leng = d3.select(this).node().getComputedTextLength(),\n        x = moveXBy(d, i, horizontalQ, categoricalQ, objectSize),\n        y = moveYBy(d, i, verticalQ, categoricalQ, objectSize)\n        // on recall, rect changes because of rotation so need Math.min(rect.height, rect.width)\n\n        var s = Math.sin(tickLabelRotation) * leng * 0\n\n        if (orient == 'top') {\n          y = -(tickLength+tickTickLabelSpacer);\n          // y = tickLength+tickTickLabelSpacer;\n\n          // y -= Math.max(rect.height, rect.width);\n          x += Math.min(rect.height, rect.width) * 0.25\n          // x -= leng * 0.25 + s\n        }\n        if (orient == 'bottom') {\n          y = tickLength+tickTickLabelSpacer;\n          x += Math.min(rect.height, rect.width) * 0.25\n          // x += leng * 0.25 - s\n        }\n        if (orient == 'left') {\n          x -= (tickLength+tickTickLabelSpacer);\n          // y += rect.height * 0.5; y-= rect.height/4\n          y += Math.min(rect.height, rect.width) * 0.25\n          // y += leng * 0.25\n        }\n        if (orient == 'right') {\n          x += (tickLength+tickTickLabelSpacer);\n          // y += Math.min(rect.height, rect.width) * 0.25\n          // y += leng * 0.25\n          y += rect.height * 0.5; y-= rect.height/4\n        }\n\n        var\n        t = 'translate('+x+','+y+')',\n        r = 'rotate('+tickLabelRotation+')'\n        return t + r\n      })\n      .on('mousemove', labelHover)\n      .on('mouseout', labelHoverOff)\n      .on('click', tickLabelOnClick)\n      // .attr('clip-path', 'url(#'+hypenate(namespace,'tick-label-clip-path')+')')\n\n      // add guidlines as needed\n       if (guideLinesQ) {\n         var gline = safeSelect(that, 'line', hypenate(namespace, 'guideline'))\n         .transition().duration(transitionDuration).ease(easeFunc)\n         .attr(\"x1\", 0)\n         .attr(\"x2\", horizontalQ ? 0 : orient == \"left\" ? guidelineSpace : -guidelineSpace)\n         .attr(\"y1\", 0)\n         .attr('y2',  verticalQ ? 0 : orient == \"top\" ? guidelineSpace : -guidelineSpace)\n         .attr('transform', function(d, i) {\n           var\n           x = moveXBy(d, i, horizontalQ, categoricalQ, objectSize),\n           y = moveYBy(d, i, verticalQ, categoricalQ, objectSize),\n           t = 'translate('+x+','+y+')'\n           return t\n         })\n       } else { that.select('line.'+hypenate(namespace, 'guideline')).remove() }\n\n    })\n\n    // apply alternating guidline thickness\n    if (guideLinesQ) {\n      container.selectAll('.'+hypenate(namespace,'guideline'))\n      .attr('stroke', function(d, i){\n        if (i % 2 == 0) { return modifyHexidecimalColorLuminance(guideLineStroke, 0.8) }\n        return guideLineStroke\n      })\n      .attr('stroke-width', function(d, i){\n        if (i % 2 == 0) { return guideLineStrokeWidth *0.8}\n        return guideLineStrokeWidth\n      })\n      .attr('minor', function(d, i){return i%2 == 0})\n    }\n\n\n    /***************************************************************************\n    ** Make the line of the axis\n    ***************************************************************************/\n    var line = safeSelect(selection, 'path', hypenate(namespace,'line'))\n    // .attr('x1', 0)\n    // .attr('x2', horizontalQ ? spaceX : 0)\n    // .attr('y1', 0)\n    // .attr('y2', horizontalQ ? 0 : spaceY)\n    .attr('d',\n      horizontalQ\n      ? 'M 0,0 H' + spaceX + ',0'\n      : 'M 0,0 V 0,' + spaceY\n    )\n    .attr('stroke', lineStroke)\n    .attr('stroke-width', lineStrokeWidth)\n    .classed('axis-line', true)\n\n\n  }\n\n  // hover of label show full text label in case it is truncated\n  function labelHover(d, i){\n    var t = d3.select(this).style('fill', 'red')\n    d3.select(t.node().parentNode).select(\"line.\"+hypenate(namespace,'tick'))\n    .attr(\"stroke\", 'red')\n    .attr(\"stroke-width\", tickStrokeWidth*2)\n\n    if (guideLinesQ) {\n      d3.select(t.node().parentNode).select('line.'+hypenate(namespace, 'guideline'))\n      .attr('stroke', 'red')\n      .attr('stroke-width', guideLineStrokeWidth*2)\n    }\n\n    var s = typeof d == 'number' ? round(d, roundTo) : d\n\n    var m = d3.mouse(d3.select('html').node())\n    var div = safeSelect(d3.select('body'), 'div', hypenate(namespace,'guideline-tooltip'))\n    .attr('id', hypenate(namespace,'guideline-tooltip'))\n    .style('position', 'absolute')\n    .style('left', (d3.event.pageX+15)+'px')\n    .style('top', (d3.event.pageY+15)+'px')\n    .style('background-color', 'white')\n    .style('border-color', 'black')\n    // .style('min-width', (tickLabelFontSize * (String(s).split('.')[0].length+3))+'px')\n    // .style('min-height', (tickLabelFontSize * (String(s).split('.')[0].length+3))+'px')\n    .style('border-radius', '10px')\n    .style('display', 'flex')\n    .style('justify-content', 'center')\n    .style('text-align', 'middle')\n    .style('padding', 4+\"px\")\n\n    .style('border-style', 'solid')\n    .style('border-width', 2)\n\n    var text = safeSelect(div, 'div')\n    .text(tickLabelOnHoverFunc(s, i))\n    .style('color', 'black')\n    .style('align-self', 'center')\n\n    var bbox = div.node().getBoundingClientRect()\n    if (bbox.x + bbox.width > window.innerWidth) {\n      div.style('left', (d3.event.pageX-15-300)+'px')\n    }\n  }\n\n  function labelHoverOff(d, i){\n    var t = d3.select(this).style('fill', 'black')\n    d3.select(t.node().parentNode).select(\"line.\"+hypenate(namespace,'tick'))\n    .attr(\"stroke\", tickStroke)\n    .attr(\"stroke-width\", tickStrokeWidth)\n\n    if (guideLinesQ) {\n      var gline = d3.select(t.node().parentNode).select('line.'+hypenate(namespace, 'guideline'))\n      var minorQ = gline.attr('minor')\n      gline.attr('stroke', function(d, ii){\n        if (minorQ == 'true') { return modifyHexidecimalColorLuminance(guideLineStroke, 0.8) }\n        return guideLineStroke\n      })\n      .attr('stroke-width', function(d, ii){\n        if (minorQ == 'true') { return guideLineStrokeWidth *0.8}\n        return guideLineStrokeWidth\n      })\n    }\n    d3.select(\"#\"+hypenate(namespace,'guideline-tooltip')).remove()\n  }\n\n\n\n  return axis\n}\n","import {hypenate, safeSelect} from './helpers';\nimport {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer} from './utils';\nimport {unique, flatten} from './array-functions';\nimport {groupingSpacer} from './grouping-spacer';\nimport {colorFunction as CF} from './color-function';\nimport {tooltip as TTip} from './tooltip';\n/*******************************************************************************\n**                                                                            **\n**                                                                            **\n**                                   BAR                                      **\n**                                                                            **\n**                                                                            **\n*******************************************************************************/\n\n/**\n * Creates a bar\n *\n * {@link https://sumneuron.gitlab.io/d3sm/demos/bar-chart-same-data-complex-grouping/index.html Demo}\n * @constructor bar\n * @param {d3.selection} selection\n * @namespace bar\n * @returns {function} bar\n */\nexport function bar ( selection ) {\n  /*\n  Assumes that data is list an object.\n\n  The keys of data will be bound to the bars.\n  The valueExtractor function will extract the value from data[key].\n\n  Grouping can be used if desired. It should be an arbitrary complex list where\n  the values are strings matching keys in data.\n  */\n  var\n  /**\n  * Data to plot. Assumed to be a object, where each key corresponds to a bar\n  * (see {@link bar#data})\n  * @param {Object} [data=undefined]\n  * @memberof bar#\n  * @property\n  */\n  data,\n  /**\n  * Which direction to render the bars in\n  * (see {@link bar#orient})\n  * @param {number} [orient='horizontal']\n  * @memberof bar#\n  * @property\n  */\n  orient='horizontal',\n  /**\n  * Amount of horizontal space (in pixels) avaible to render the bar in\n  * (see {@link bar#spaceX})\n  * @param {number} [spaceX=undefined]\n  * @memberof bar#\n  * @property\n  */\n  spaceX,\n  /**\n  * Amount of vertical space (in pixels) avaible to render the bar in\n  * (see {@link bar.spaceY})\n  * @param {number} [spaceY=undefined]\n  * @memberof bar#\n  * @property\n  */\n  spaceY,\n\n  /**\n  * Whether or not to allow bar to render elements pass the main spatial dimension\n  * given the orientation (see {@link bar#orient}), where {@link bar#orient}=\"horizontal\"\n  * the main dimension is {@link bar#spaceX} and where {@link bar#orient}=\"vertical\"\n  * the main dimension is {@link bar#spaceY}\n  * @param {boolean} [overflowQ=false]\n  * @memberof bar#\n  * @property\n  */\n  overflowQ = false,\n\n  /**\n  * An array - putatively of other arrays - depicting how bars should be arranged\n  * @param {Array[]} [grouping=undefined]\n  * @memberof bar#\n  * @property\n  */\n  grouping,\n\n  /**\n  * How to get the value of the bar\n  * @param {function} [valueExtractor=function(key, index) { return data[key] }]\n  * @memberof bar#\n  * @property\n  */\n  valueExtractor = function(key, index) { return data[key] },\n  /**\n  * How to sort the bars - if {@link bar#grouping} is not provided.\n  * @param {function} [sortingFunction=function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}]\n  * @memberof bar#\n  * @property\n  */\n  sortingFunction = function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])},\n\n  /**\n  * The scale for which bar values should be transformed by\n  * @param {d3.scale} [scale=d3.scaleLinear]\n  * @memberof bar#\n  * @property\n  */\n  scale = d3.scaleLinear(),\n  /**\n  * The padding for the domain of the scale (see {@link bar#scale})\n  * @param {number} [domainPadding=0.5]\n  * @memberof bar#\n  * @property\n  */\n  domainPadding = 0.5,\n\n  /**\n  * Default space for the spacer (percentage) of main dimension given the orientation\n  * (see {@link bar#orient}), where {@link bar#orient}=\"horizontal\"\n  * the main dimension is {@link bar#spaceX} and where {@link bar#orient}=\"vertical\"\n  * the main dimension is {@link bar#spaceY} between bars\n  * @param {number} [objectSpacer=0.05]\n  * @memberof bar#\n  * @property\n  */\n  objectSpacer = 0.05,\n  /**\n  * The minimum size that an object can be\n  * @param {number} [minObjectSize=50]\n  * @memberof bar#\n  * @property\n  */\n  minObjectSize = 50,\n  /**\n  * The maximum size that an object can be\n  * @param {number} [maxObjectSize=100]\n  * @memberof bar#\n  * @property\n  */\n  maxObjectSize = 100,\n\n  /**\n  * The stroke width of the bars\n  * @param {number} [barStrokeWidth=2]\n  * @memberof bar#\n  * @property\n  */\n  barStrokeWidth = 2,\n  /**\n  * Instance of ColorFunction\n  * @param {function} [colorFunction = colorFunction()]\n  * @memberof bar#\n  * @property\n  */\n  colorFunction = CF(),\n\n\n  /**\n  * Color of the background\n  * @param {string} [backgroundFill=\"transparent\"]\n  * @memberof bar#\n  * @property\n  */\n  backgroundFill = 'transparent',\n  /**\n  * Namespace for all items made by this instance of bar\n  * @param {string} [namespace=\"d3sm-bar\"]\n  * @memberof bar#\n  * @property\n  */\n  namespace = 'd3sm-bar',\n  /**\n  * Class name for bar container (<g> element)\n  * @param {string} [objectClass=\"bar\"]\n  * @memberof bar#\n  * @property\n  */\n  objectClass = 'bar',\n\n  /**\n  * Duration of all transitions of this element\n  * @param {number} [transitionDuration=1000]\n  * @memberof bar#\n  * @property\n  */\n  transitionDuration = 1000,\n  /**\n  * Easing function for transitions\n  * @param {d3.ease} [easeFunc=d3.easeExp]\n  * @memberof bar#\n  * @property\n  */\n  easeFunc = d3.easeExp,\n\n  // useful values to extract to prevent re-calculation\n  /**\n  * The keys of the bars\n  * @param {string[]} [barKeys=undefined]\n  * @memberof bar#\n  * @property\n  */\n  barKeys,\n  /**\n  * The values of the bars\n  * @param {number[]} [barValues=undefined]\n  * @memberof bar#\n  * @property\n  */\n  barValues,\n  /**\n  * The objectSize (actual width) used by the bars\n  * @param {number} [objectSize=undefined]\n  * @memberof bar#\n  * @property\n  */\n  objectSize,\n  /**\n  * The spacerSize (actual width) used by the spacers between the bars\n  * @param {number} [spacerSize=undefined]\n  * @memberof bar#\n  * @property\n  */\n  spacerSize,\n  /**\n  * Instance of Tooltip\n  * @param {function} [tooltip=tooltip()]\n  * @memberof bar#\n  * @property\n  */\n  tooltip = TTip(),\n  barPercent = 1\n\n  /**\n   * Gets or sets the selection in which items are manipulated\n   * @param {d3.selection} [_=none]\n   * @returns {bar | d3.selection}\n   * @memberof bar\n   * @property\n   * by default selection = selection\n   */\n  bar.selection = function(_) { return arguments.length ? (selection = _, bar) : selection; };\n  /**\n   * Gets or sets the data\n   * (see {@link bar#data})\n   * @param {number} [_=none]\n   * @returns {bar | object}\n   * @memberof bar\n   * @property\n   */\n  bar.data = function(_) { return arguments.length ? (data = _, bar) : data; };\n  /**\n   * Gets or sets the orient of the bars\n   * (see {@link bar#orient})\n   * @param {number} [_=none]\n   * @returns {bar | object}\n   * @memberof bar\n   * @property\n   */\n  bar.orient = function(_) { return arguments.length ? (orient = _, bar) : orient; };\n  /**\n   * Gets or sets the amount of horizontal space in which items are manipulated\n   * (see {@link bar#spaceX})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {bar | number}\n   * @memberof bar\n   * @property\n   * by default spaceX = undefined\n   */\n  bar.spaceX = function(_) { return arguments.length ? (spaceX = _, bar) : spaceX; };\n  /**\n   * Gets or sets the amount of vertical space in which items are manipulated\n   * (see {@link bar#spaceY})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {bar | number}\n   * @memberof bar\n   * @property\n   * by default spaceY = undefined\n   */\n  bar.spaceY = function(_) { return arguments.length ? (spaceY = _, bar) : spaceY; };\n\n  /**\n   * Gets / sets whether or not bar is allowed to go beyond specified dimensions\n   * (see {@link bar#spaceX})\n   * @param {boolean} [_=none]\n   * @returns {bar | boolean}\n   * @memberof bar\n   * @property\n   * by default overflowQ = false\n   */\n  bar.overflowQ = function(_) { return arguments.length ? (overflowQ = _, bar) : overflowQ; };\n  /**\n   * Gets / sets the grouping of the bars\n   * (see {@link bar#grouping})\n   * @param {Array[]} [_=none]\n   * @returns {bar | Array[]}\n   * @memberof bar\n   * @property\n   * by default grouping = undefined\n   */\n  bar.grouping = function(_) { return arguments.length ? (grouping = _, bar) : grouping; };\n  /**\n   * Gets / sets the valueExtractor\n   * (see {@link bar#valueExtractor})\n   * @param {function} [_=none]\n   * @returns {bar | function}\n   * @memberof bar\n   * @property\n   * by default valueExtractor = function(key, index) { return data[key] },\n   */\n  bar.valueExtractor = function(_) { return arguments.length ? (valueExtractor = _, bar) : valueExtractor; };\n  /**\n   * Gets / sets the sortingFunction\n   * (see {@link bar#sortingFunction})\n   * @param {function} [_=none]\n   * @returns {bar | function}\n   * @memberof bar\n   * @property\n   * by default sortingFunction = function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])},\n   */\n  bar.sortingFunction = function(_) { return arguments.length ? (sortingFunction = _, bar) : sortingFunction; };\n  /**\n   * Gets / sets the scale for which the bar values should be transformed by\n   * (see {@link bar#scale})\n   * @param {d3.scale} [_=none]\n   * @returns {bar | d3.scale}\n   * @memberof bar\n   * @property\n   * by default scale = d3.scaleLinear()\n   */\n  bar.scale = function(_) { return arguments.length ? (scale = _, bar) : scale; };\n  /**\n   * Gets / sets the padding for the domain of the scale\n   * (see {@link bar#domainPadding})\n   * @param {number} [_=none]\n   * @returns {bar | number}\n   * @memberof bar\n   * @property\n   * by default domainPadding = 0.5\n   */\n  bar.domainPadding = function(_) { return arguments.length ? (domainPadding = _, bar) : domainPadding; };\n  /**\n   * Gets / sets objectSpacer\n   * (see {@link bar#objectSpacer})\n   * @param {number} [_=none]\n   * @returns {bar | number}\n   * @memberof bar\n   * @property\n   * by default objectSpacer = 0.05\n   */\n  bar.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, bar) : objectSpacer; };\n  /**\n   * Gets / sets the minObjectSize\n   * (see {@link bar#minObjectSize})\n   * @param {number} [_=none]\n   * @returns {bar | number}\n   * @memberof bar\n   * @property\n   * by default minObjectSize = 50\n   */\n  bar.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, bar) : minObjectSize; };\n  /**\n   * Gets / sets the maxObjectSize\n   * (see {@link bar#maxObjectSize})\n   * @param {number} [_=none]\n   * @returns {bar | number}\n   * @memberof bar\n   * @property\n   * by default maxObjectSize = 100\n   */\n  bar.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, bar) : maxObjectSize; };\n\n  /**\n   * Gets / sets the barStrokeWidth\n   * (see {@link bar#barStrokeWidth})\n   * @param {number} [_=none]\n   * @returns {bar | number}\n   * @memberof bar\n   * @property\n   * by default barStrokeWidth = 2\n   */\n  bar.barStrokeWidth = function(_) { return arguments.length ? (barStrokeWidth = _, bar) : barStrokeWidth; };\n  /**\n   * Gets / sets the colorFunction\n   * (see {@link bar#colorFunction})\n   * @param {number} [_=none]\n   * @returns {bar | number}\n   * @memberof bar\n   * @property\n   * by default colorFunction = colorFunction()\n   */\n  bar.colorFunction = function(_) { return arguments.length ? (colorFunction = _, bar) : colorFunction; };\n\n  /**\n   * Gets / sets the backgroundFill\n   * (see {@link bar#backgroundFill})\n   * @param {string} [_=none]\n   * @returns {bar | string}\n   * @memberof bar\n   * @property\n   * by default backgroundFill = 'transparent'\n   */\n  bar.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, bar) : backgroundFill; };\n  /**\n   * Gets / sets the namespace\n   * (see {@link bar#namespace})\n   * @param {string} [_=none]\n   * @returns {bar | string}\n   * @memberof bar\n   * @property\n   * by default namespace = 'd3sm-bar'\n   */\n  bar.namespace = function(_) { return arguments.length ? (namespace = _, bar) : namespace; };\n  /**\n   * Gets / sets the objectClass\n   * (see {@link bar#objectClass})\n   * @param {string} [_=none]\n   * @returns {bar | string}\n   * @memberof bar\n   * @property\n   * by default objectClass = 'tick-group'\n   */\n  bar.objectClass = function(_) { return arguments.length ? (objectClass = _, bar) : objectClass; };\n  /**\n   * Gets / sets the transitionDuration\n   * (see {@link bar#transitionDuration})\n   * @param {number} [_=none]\n   * @returns {bar | number}\n   * @memberof bar\n   * @property\n   * by default transitionDuration = 1000\n   */\n  bar.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, bar) : transitionDuration; };\n  /**\n   * Gets / sets the easeFunc\n   * (see {@link bar#easeFunc})\n   * @param {d3.ease} [_=none]\n   * @returns {bar | d3.ease}\n   * @memberof bar\n   * @property\n   * by default easeFunc = d3.easeExp\n   */\n  bar.easeFunc = function(_) { return arguments.length ? (easeFunc = _, bar) : easeFunc; };\n\n\n  /**\n   * Gets / sets the barKeys\n   * (see {@link bar#barKeys})\n   * @param {string[]} [_=none]\n   * @returns {bar | string[]}\n   * @memberof bar\n   * @property\n   * by default barKeys = undefined\n   */\n  bar.barKeys = function(_) { return arguments.length ? (barKeys = _, bar) : barKeys; };\n  /**\n   * Gets / sets the barValues\n   * (see {@link bar#barValues})\n   * @param {number[]} [_=none]\n   * @returns {bar | number[]}\n   * @memberof bar\n   * @property\n   * by default barValues = undefined\n   */\n  bar.barValues = function(_) { return arguments.length ? (barValues = _, bar) : barValues; };\n  /**\n   * Gets / sets the objectSize\n   * (see {@link bar#objectSize})\n   * @param {number} [_=none]\n   * @returns {bar | number}\n   * @memberof bar\n   * @property\n   * by default objectSize = undefined\n   */\n  bar.objectSize = function(_) { return arguments.length ? (objectSize = _, bar) : objectSize; };\n  /**\n   * Gets / sets the spacerSize\n   * (see {@link bar#spacerSize})\n   * @param {number} [_=none]\n   * @returns {bar | number}\n   * @memberof bar\n   * @property\n   * by default spacerSize = undefined\n   */\n  bar.spacerSize = function(_) { return arguments.length ? (spacerSize = _, bar) : spacerSize; };\n\n  /**\n   * Gets / sets the tooltip\n   * (see {@link bar#tooltip})\n   * @param {tooltip} [_=none]\n   * @returns {bar | tooltip}\n   * @memberof bar\n   * @property\n   * by default tooltip = tooltip()\n   */\n  bar.tooltip = function(_) { return arguments.length ? (tooltip = _, bar) : tooltip; };\n\n  bar.barPercent = function(_) { return arguments.length ? (barPercent = _, bar) : barPercent; };\n\n  function bar() {\n    // for convenience in handling orientation specific values\n    var horizontalQ = (orient == 'horizontal' || orient == 'bottom' || orient == 'top') ? true : false\n    var verticalQ = !horizontalQ\n\n    // background cliping rectangle\n    var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}\n    var container = setupContainer( selection, namespace, bgcpRect, backgroundFill );\n\n    // to prevent re-calculation and getters to be passed to axes\n    barKeys = d3.keys(data)\n    barValues = barKeys.map(valueExtractor)\n\n    // if grouping is undefined sort barKeys by sortingFunction\n    var ordered = (grouping == undefined) ? barKeys.sort(sortingFunction) : grouping\n    // ordered might be nested depending on grouping\n    barKeys = flatten(ordered)\n\n    var numberOfObjects = barKeys.length\n    var extent = [Math.min(...barValues) - domainPadding,Math.max(...barValues) + domainPadding];\n\n\n\n    // set the scale\n\n    scale.domain(extent).range(horizontalQ\n      ? [0,spaceY]\n      : orient == 'right'\n        ? [0, spaceX]\n        : [spaceX, 0]\n    )\n    var space = horizontalQ ? spaceX : spaceY\n    // calculate object size\n    objectSize =  (objectSize == undefined)\n    ? calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ)\n    : objectSize\n\n    // calculate spacer size if needed\n    spacerSize = (spacerSize == undefined)\n    ? calculateWidthOfSpacer(barKeys, space, objectSize, numberOfObjects, objectSpacer, overflowQ)\n    : spacerSize\n    // make the nested groups\n    var spacerFunction = groupingSpacer()\n    .horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects)\n    .objectClass(objectClass).objectSize(objectSize).spacerSize(spacerSize)\n    .transitionDuration(transitionDuration).easeFunc(easeFunc)\n    .namespace(namespace)\n    // safe default function\n    var defaultExit = spacerFunction.exitFunction()\n\n    spacerFunction.exitFunction(function(sel){\n      // use default to move objects off screen\n      // console.log(\"EXIT\", sel.nodes(), objectSize, scale(extent[1]))\n      if (objectSize == undefined) {console.log(sel.nodes(), objectSize)}\n      defaultExit(sel)\n      sel.selectAll('g').classed(\"to-remove\", true)\n      // shrink rectangles in addition\n      sel.selectAll('* > rect')\n      .transition().duration(transitionDuration)\n      .attr('transform', function(d, i) {\n        var\n        x = horizontalQ\n          ? 0\n          : 0\n        ,\n        y = verticalQ\n          ? 0\n          : scale(extent[1])\n        ,\n        t = 'translate('+x+','+y+')'\n        return t\n      })\n      .attr('width', horizontalQ ? objectSize : 0)\n      .attr('height', verticalQ ? objectSize : 0)\n      .remove()\n    })\n\n\n    // move stuff\n    spacerFunction(container, ordered, 0)\n\n\n\n\n\n    var parentIndexArray = []\n    container.selectAll('g:not(.to-remove).'+objectClass)\n    .each(function(d, i){parentIndexArray.push(Number(d3.select(this).attr('parent-index')))})\n\n\n    colorFunction = colorFunction.colorBy() == 'index'\n    ? colorFunction.dataExtent([0, Math.max(...parentIndexArray)])\n    : colorFunction.dataExtent(extent)\n\n\n\n    container.selectAll('g.'+objectClass+':not(.to-remove)').each(function(key, i) {\n      // console.log(key, scale(extent[1]) - scale(valueExtractor(key, i)))\n      var t = d3.select(this),\n      currentData = data[key],\n      value = valueExtractor(key, i),\n      i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'),\n      fillColor = colorFunction(key, value, i, 'fill'), // prevent duplicate computation\n      strokeColor = colorFunction(key, value, i,  'stroke')\n\n\n      var bar = safeSelect(t, 'rect', 'bar-rect')\n\n      if (bar.attr('transform') == undefined) {\n        bar.attr('transform', function(d, i) {\n          var\n          x = horizontalQ\n            ? 0\n            : 0\n          ,\n          y = verticalQ\n            ? 0\n            : scale(extent[1])\n          ,\n          t = 'translate('+x+','+y+')'\n          return t\n        })\n        .attr('width', horizontalQ ? objectSize : 0)\n        .attr('height', verticalQ ? objectSize : 0)\n\n      }\n\n\n      bar.transition().duration(transitionDuration).ease(easeFunc)\n      .attr('transform', function(d, i) {\n        var\n        x = horizontalQ\n          ? objectSize - objectSize * barPercent\n          : orient == 'right'\n            ? scale(extent[1]) - scale(value)\n            : objectSize - objectSize * barPercent\n          ,\n        y = verticalQ\n          ? objectSize - objectSize * barPercent\n          : scale(extent[1]) - scale(value)\n        ,\n        t = 'translate('+x+','+y+')'\n        return t\n      })\n      .attr('width', horizontalQ ? objectSize * barPercent : scale(value))\n      .attr('height', verticalQ ? objectSize * barPercent: scale(value))\n      .attr('fill', fillColor)\n      .attr('stroke', strokeColor)\n      .attr('stroke-width', barStrokeWidth)\n\n\n\n      t.on('mouseover', function(d, i){\n        container.selectAll('g.'+objectClass).style('opacity', 0.2)\n        t.style('opacity', 1)\n        bar.attr('stroke-width',barStrokeWidth*2)\n\n      })\n      t.on('mouseout', function(){\n        container.selectAll('g.'+objectClass).style('opacity', 1)\n        bar.attr('stroke-width', barStrokeWidth)\n      })\n    })\n\n    tooltip.selection(container.selectAll('.bar-rect'))\n    .data(data)\n\n    tooltip()\n\n  }\n  return bar\n}\n","import {hypenate, safeSelect} from './helpers';\nimport {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer, log} from './utils';\nimport {unique} from './array-functions';\nimport {groupingSpacer} from './grouping-spacer';\nimport {colorFunction as CF} from './color-function';\nimport {tooltip as TTip} from './tooltip';\n\n\n/**\n * Creates a bubbleHeatmap\n *\n * {@link https://sumneuron.gitlab.io/d3sm/demos/bubble-heatmap/index.html Demo}\n * @constructor bubbleHeatmap\n * @param {d3.selection} selection\n * @namespace bubbleHeatmap\n * @returns {function} bubbleHeatmap\n */\nfunction bubbleHeatmap( selection ) {\n  var\n  /**\n  * Data to plot. Assumed to be a object, where each key corresponds to a cell\n  * (see {@link bubbleHeatmap#data})\n  * @param {Object} [data=undefined]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  data,\n\n  /**\n  * Amount of horizontal space (in pixels) avaible to render the bubbleHeatmap in\n  * (see {@link bubbleHeatmap#spaceX})\n  * @param {number} [spaceX=undefined]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  spaceX,\n  /**\n  * Amount of vertical space (in pixels) avaible to render the bubbleHeatmap in\n  * (see {@link bubbleHeatmap.spaceY})\n  * @param {number} [spaceY=undefined]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  spaceY,\n\n  /**\n  * The internal key of the cell specifiying to which x axis key it belongs\n  * (see {@link bubbleHeatmap.xKey})\n  * @param {string} [xKey='x']\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  xKey = 'x',\n  /**\n  * The internal key of the cell specifiying to which y axis key it belongs\n  * (see {@link bubbleHeatmap.yKey})\n  * @param {string} [yKey='y']\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  yKey = 'y',\n  /**\n  * The internal key of the cell specifiying what value to use to determine the radius\n  * (see {@link bubbleHeatmap.rKey})\n  * @param {string} [rKey='r']\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  rKey = 'r',\n  /**\n  * The internal key of the cell specifiying what value to use to determine the color\n  * (see {@link bubbleHeatmap.vKey})\n  * @param {string} [vKey='v']\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  vKey = 'v',\n\n  /**\n  * Function for extracting the the value from xKey.\n  * (see {@link bubbleHeatmap.xExtractor})\n  * @param {function} [xExtractor=function(key, i) { return data[key][xKey] }]\n  * @returns {string}\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  xExtractor = function(key, i) {return data[key][xKey] },\n  /**\n  * Function for extracting the the value from yKey.\n  * (see {@link bubbleHeatmap.yExtractor})\n  * @param {function} [yExtractor=function(key, i) { return data[key][yKey] }]\n  * @returns {string}\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  yExtractor = function(key, i) { return data[key][yKey] },\n  /**\n  * Function for extracting the the value from rKey.\n  * (see {@link bubbleHeatmap.rExtractor})\n  * @param {function} [rExtractor=function(key, i) { return data[key][rKey] }]\n  * @returns {number}\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  rExtractor = function(key, i) { return data[key][rKey] },\n  /**\n  * Function for extracting the the value from vKey.\n  * (see {@link bubbleHeatmap.vExtractor})\n  * @param {function} [vExtractor=function(key, i) { return data[key][vKey] }]\n  * @returns {number}\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  vExtractor = function(key, i) { return data[key][vKey] },\n\n\n  /**\n  * Whether or not to allow bubbleHeatmap to render elements pass the main spatial dimension\n  * given the orientation (see {@link bubbleHeatmap#orient}), where {@link bubbleHeatmap#orient}=\"bottom\" or {@link bubbleHeatmap#orient}=\"top\"\n  * the main dimension is {@link bubbleHeatmap#spaceX} and where {@link bubbleHeatmap#orient}=\"left\" or {@link bubbleHeatmap#orient}=\"right\"\n  * the main dimension is {@link bubbleHeatmap#spaceY}\n  * @param {boolean} [overflowQ=false]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  overflowQ = false,\n\n  /**\n  * The scale for which the radius values should be transformed by\n  * @param {d3.scale} [scale=d3.scaleLinear]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  scale = d3.scaleLinear(),\n  /**\n  * The padding for the domain of the scale (see {@link bubbleHeatmap#scale})\n  * @param {number} [domainPadding=0.5]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  domainPadding = 0.5,\n\n  /**\n  * Default space for the spacer (percentage) of main dimension given the orientation\n  * (see {@link bubbleHeatmap#orient}), where {@link bubbleHeatmap#orient}=\"horizontal\"\n  * the main dimension is {@link bubbleHeatmap#spaceX} and where {@link bubbleHeatmap#orient}=\"vertical\"\n  * the main dimension is {@link bubbleHeatmap#spaceY} between bubbles\n  * @param {number} [objectSpacer=0.0]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  objectSpacer = 0.0,\n  /**\n  * The minimum size that an object can be\n  * @param {number} [minObjectSize=50]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  minObjectSize = 50,\n  /**\n  * The maximum size that an object can be\n  * @param {number} [maxObjectSize=100]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  maxObjectSize = 100,\n\n\n  /**\n  * The stroke width of the bubbles\n  * @param {number} [bubbleStrokeWidth=2]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  bubbleStrokeWidth = 2,\n  // colorFunc = colorFunction(),\n\n  /**\n  * Color of the background\n  * @param {string} [backgroundFill=\"transparent\"]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  backgroundFill = 'transparent',\n  /**\n  * Namespace for all items made by this instance of bubbleHeatmap\n  * @param {string} [namespace=\"d3sm-bubble\"]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  namespace = 'd3sm-bubble',\n  /**\n  * Class name for bubble container (<g> element)\n  * @param {string} [objectClass=\"bubble\"]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  objectClass = 'bubble',\n  /**\n  * Duration of all transitions of this element\n  * @param {number} [transitionDuration=1000]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  transitionDuration = 1000,\n  /**\n  * Easing function for transitions\n  * @param {d3.ease} [easeFunc=d3.easeExp]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  easeFunc = d3.easeExp,\n\n  /**\n  * Stores the keys of all the cells\n  * Calculated after bubbleHeatmap called.\n  * @param {string[]} [cellKeys=undefined]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  cellKeys,\n  /**\n  * Stores the list of unique xValues\n  * Calculated after bubbleHeatmap called.\n  * @param {string[]} [xValues=undefined]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  xValues,\n  /**\n  * Stores the list of unique yValues\n  * Calculated after bubbleHeatmap called.\n  * @param {string[]} [yValues=undefined]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  yValues,\n  /**\n  * Stores the list of unique rValues\n  * Calculated after bubbleHeatmap called.\n  * @param {string[]} [rValues=undefined]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  rValues,\n  /**\n  * Stores the list of unique vValues\n  * Calculated after bubbleHeatmap called.\n  * @param {string[]} [vValues=undefined]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  vValues,\n\n  xKeySortingFunction = function(a, b) { return xExtractor(a) - xExtractor(b) },\n  yKeySortingFunction = function(a, b) { return yExtractor(a) - yExtractor(b) },\n  rKeySortingFunction = function(a, b) { return rExtractor(a) - rExtractor(b) },\n  vKeySortingFunction = function(a, b) { return vExtractor(a) - vExtractor(b) },\n\n  /**\n  * Instance of ColorFunction with .colorBy set to 'category'\n  * @function colorFunction\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  colorFunction = CF().colorBy('value'),\n  /**\n  * Instance of Tooltip\n  * @function tooltip\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  tooltip = TTip(),\n\n  /**\n  * store the size the bubble could be in the x dimension\n  * the actuall size of the bubble will be the min of xSize and {@link bubbleHeatmap#ySize}\n  * Calculated after bubbleHeatmap called.\n  * @param {string[]} [xSize=undefined]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  xSize,\n  /**\n  * store the size of the spacer in the x dimension\n  * Calculated after bubbleHeatmap called.\n  * @param {string[]} [xSpacerSize=undefined]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  xSpacerSize,\n\n  /**\n  * store the size the bubble could be in the y dimension\n  * the actuall size of the bubble will be the min of xSize and {@link bubbleHeatmap#xSize}\n  * Calculated after bubbleHeatmap called.\n  * @param {string[]} [ySize=undefined]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  ySize,\n  /**\n  * store the size of the spacer in the y dimension.\n  * Calculated after bubbleHeatmap called.\n  * @param {string[]} [xSpacerSize=undefined]\n  * @memberof bubbleHeatmap#\n  * @property\n  */\n  ySpacerSize\n\n  /**\n   * Gets or sets the selection in which items are manipulated\n   * @param {d3.selection} [_=none]\n   * @returns {bubbleHeatmap | d3.selection}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default selection = selection\n   */\n  bhm.selection = function(_) { return arguments.length ? (selection = _, bhm) : selection; }\n  /**\n   * Gets or sets the data\n   * (see {@link bubbleHeatmap#data})\n   * @param {number} [_=none]\n   * @returns {bubbleHeatmap | object}\n   * @memberof bubbleHeatmap\n   * @property\n   */\n  bhm.data = function(_) { return arguments.length ? (data = _, bhm) : data; }\n  // bhm.orient = function(_) { return arguments.length ? (orient = _, bhm) : orient; }\n  /**\n   * Gets or sets the amount of horizontal space in which items are manipulated\n   * (see {@link bubbleHeatmap#spaceX})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {bubbleHeatmap | number}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default spaceX = undefined\n   */\n  bhm.spaceX = function(_) { return arguments.length ? (spaceX = _, bhm) : spaceX; }\n  /**\n   * Gets or sets the amount of vertical space in which items are manipulated\n   * (see {@link bubbleHeatmap#spaceY})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {bubbleHeatmap | number}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default spaceY = undefined\n   */\n  bhm.spaceY = function(_) { return arguments.length ? (spaceY = _, bhm) : spaceY; }\n\n  /**\n   * Gets or sets the xKey\n   * (see {@link bubbleHeatmap#xKey})\n   * @param {string} [_=none]\n   * @returns {bubbleHeatmap | string}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default xKey = 'x'\n   */\n  bhm.xKey = function(_) { return arguments.length ? (xKey = _, bhm) : xKey; }\n  /**\n   * Gets or sets the yKey\n   * (see {@link bubbleHeatmap#yKey})\n   * @param {string} [_=none]\n   * @returns {bubbleHeatmap | string}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default yKey = 'y'\n   */\n  bhm.yKey = function(_) { return arguments.length ? (yKey = _, bhm) : yKey; }\n  /**\n   * Gets or sets the rKey\n   * (see {@link bubbleHeatmap#rKey})\n   * @param {string} [_=none]\n   * @returns {bubbleHeatmap | string}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default rKey = 'r'\n   */\n  bhm.rKey = function(_) { return arguments.length ? (rKey = _, bhm) : rKey; }\n  /**\n   * Gets or sets the vKey\n   * (see {@link bubbleHeatmap#vKey})\n   * @param {string} [_=none]\n   * @returns {bubbleHeatmap | string}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default vKey = 'y'\n   */\n  bhm.vKey = function(_) { return arguments.length ? (vKey = _, bhm) : vKey; }\n\n  /**\n   * Gets or sets the cellKeys\n   * (see {@link bubbleHeatmap#cellKeys})\n   * @param {string[]} [_=none]\n   * @returns {bubbleHeatmap | string[]}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default cellKeys = undefined\n   */\n  bhm.cellKeys = function(_) { return arguments.length ? (cellKeys = _, bhm) : cellKeys; }\n  /**\n   * Gets or sets the xValues\n   * (see {@link bubbleHeatmap#xValues})\n   * @param {string[]} [_=none]\n   * @returns {bubbleHeatmap | string[]}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default xValues = undefined\n   */\n  bhm.xValues = function(_) { return arguments.length ? (xValues = _, bhm) : xValues; }\n  /**\n   * Gets or sets the yValues\n   * (see {@link bubbleHeatmap#yValues})\n   * @param {string[]} [_=none]\n   * @returns {bubbleHeatmap | string[]}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default yValues = undefined\n   */\n  bhm.yValues = function(_) { return arguments.length ? (yValues = _, bhm) : yValues; }\n  /**\n   * Gets or sets the rValues\n   * (see {@link bubbleHeatmap#rValues})\n   * @param {number[]} [_=none]\n   * @returns {bubbleHeatmap | number[]}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default rValues = undefined\n   */\n  bhm.rValues = function(_) { return arguments.length ? (rValues = _, bhm) : rValues; }\n  /**\n   * Gets or sets the vValues\n   * (see {@link bubbleHeatmap#vValues})\n   * @param {number[]} [_=none]\n   * @returns {bubbleHeatmap | number[]}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default vValues = undefined\n   */\n  bhm.vValues = function(_) { return arguments.length ? (vValues = _, bhm) : vValues; }\n\n\n  /**\n   * Gets or sets the xExtractor\n   * (see {@link bubbleHeatmap#xExtractor})\n   * @param {function} [_=none]\n   * @returns {bubbleHeatmap | function}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default xExtractor = undefined\n   */\n  bhm.xExtractor = function(_) { return arguments.length ? (xExtractor = _, bhm) : xExtractor; }\n  /**\n   * Gets or sets the yExtractor\n   * (see {@link bubbleHeatmap#yExtractor})\n   * @param {function} [_=none]\n   * @returns {bubbleHeatmap | function}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default yExtractor = undefined\n   */\n  bhm.yExtractor = function(_) { return arguments.length ? (yExtractor = _, bhm) : yExtractor; }\n  /**\n   * Gets or sets the rExtractor\n   * (see {@link bubbleHeatmap#rExtractor})\n   * @param {function} [_=none]\n   * @returns {bubbleHeatmap | function}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default rExtractor = undefined\n   */\n  bhm.rExtractor = function(_) { return arguments.length ? (rExtractor = _, bhm) : rExtractor; }\n  /**\n   * Gets or sets the vExtractor\n   * (see {@link bubbleHeatmap#vExtractor})\n   * @param {function} [_=none]\n   * @returns {bubbleHeatmap | function}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default vExtractor = undefined\n   */\n  bhm.vExtractor = function(_) { return arguments.length ? (vExtractor = _, bhm) : vExtractor; }\n\n  /**\n   * Gets / sets whether or not bubbleHeatmap is allowed to go beyond specified dimensions\n   * (see {@link bubbleHeatmap#spaceX})\n   * @param {boolean} [_=none]\n   * @returns {bubbleHeatmap | boolean}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default overflowQ = false\n   */\n  bhm.overflowQ = function(_) { return arguments.length ? (overflowQ = _, bhm) : overflowQ; }\n  /**\n   * Gets / sets the scale for which the radius of bubbles should be transformed by\n   * (see {@link bubbleHeatmap#scale})\n   * @param {d3.scale} [_=none]\n   * @returns {bubbleHeatmap | d3.scale}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default scale = d3.scaleLinear()\n   */\n  bhm.scale = function(_) { return arguments.length ? (scale = _, bhm) : scale; };\n  /**\n   * Gets / sets the padding for the domain of the scale\n   * (see {@link bubbleHeatmap#domainPadding})\n   * @param {number} [_=none]\n   * @returns {bubbleHeatmap | number}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default domainPadding = 0.5\n   */\n  bhm.domainPadding = function(_) { return arguments.length ? (domainPadding = _, bhm) : domainPadding; };\n  /**\n   * Gets / sets objectSpacer\n   * (see {@link bubbleHeatmap#objectSpacer})\n   * @param {number} [_=none]\n   * @returns {bubbleHeatmap | number}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default objectSpacer = 0.0\n   */\n  bhm.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, objectSpacer) : data; }\n  /**\n   * Gets / sets the minObjectSize\n   * (see {@link bubbleHeatmap#minObjectSize})\n   * @param {number} [_=none]\n   * @returns {bubbleHeatmap | number}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default minObjectSize = 50\n   */\n  bhm.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, bhm) : minObjectSize; }\n  /**\n   * Gets / sets the maxObjectSize\n   * (see {@link bubbleHeatmap#maxObjectSize})\n   * @param {number} [_=none]\n   * @returns {bubbleHeatmap | number}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default maxObjectSize = 100\n   */\n  bhm.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, bhm) : maxObjectSize; }\n  /**\n   * Gets / sets the bubbleStrokeWidth\n   * (see {@link bubbleHeatmap#bubbleStrokeWidth})\n   * @param {number} [_=none]\n   * @returns {bubbleHeatmap | number}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default bubbleStrokeWidth = 2\n   */\n  bhm.bubbleStrokeWidth = function(_) { return arguments.length ? (bubbleStrokeWidth = _, bhm) : bubbleStrokeWidth; }\n  /**\n   * Gets / sets the backgroundFill\n   * (see {@link bubbleHeatmap#backgroundFill})\n   * @param {string} [_=none]\n   * @returns {bubbleHeatmap | string}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default backgroundFill = 'transparent'\n   */\n  bhm.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, bhm) : backgroundFill; }\n  /**\n   * Gets / sets the namespace\n   * (see {@link bubbleHeatmap#namespace})\n   * @param {string} [_=none]\n   * @returns {bubbleHeatmap | string}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default namespace = 'd3sm-bubbleHeatmap'\n   */\n  bhm.namespace = function(_) { return arguments.length ? (namespace = _, bhm) : namespace; }\n  /**\n   * Gets / sets the objectClass\n   * (see {@link bubbleHeatmap#objectClass})\n   * @param {string} [_=none]\n   * @returns {bubbleHeatmap | string}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default objectClass = 'tick-group'\n   */\n  bhm.objectClass = function(_) { return arguments.length ? (objectClass = _, bhm) : objectClass; }\n  /**\n   * Gets / sets the transitionDuration\n   * (see {@link bubbleHeatmap#transitionDuration})\n   * @param {number} [_=none]\n   * @returns {bubbleHeatmap | number}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default transitionDuration = 1000\n   */\n  bhm.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, bhm) : transitionDuration; }\n  /**\n   * Gets / sets the easeFunc\n   * (see {@link bubbleHeatmap#easeFunc})\n   * @param {d3.ease} [_=none]\n   * @returns {bubbleHeatmap | d3.ease}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default easeFunc = d3.easeExp\n   */\n  bhm.easeFunc = function(_) { return arguments.length ? (easeFunc = _, bhm) : easeFunc; }\n\n  /**\n   * Gets / sets the tooltip\n   * (see {@link bubbleHeatmap#tooltip})\n   * @param {tooltip} [_=none]\n   * @returns {bubbleHeatmap | tooltip}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default tooltip = tooltip()\n   */\n  bhm.tooltip = function(_) { return arguments.length ? (tooltip = _, bhm) : tooltip; }\n\n  /**\n   * Gets / sets the colorFunction\n   * (see {@link bubbleHeatmap#colorFunction})\n   * @param {colorFunction} [_=none]\n   * @returns {bubbleHeatmap | colorFunction}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default colorFunction = colorFunction()\n   */\n  bhm.colorFunction = function(_) { return arguments.length ? (colorFunction = _, bhm) : colorFunction; }\n\n  /**\n   * Gets / sets the xSize\n   * (see {@link bubbleHeatmap#xSize})\n   * @param {number} [_=none]\n   * @returns {bubbleHeatmap | number}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default xSize = undefined\n   */\n  bhm.xSize = function(_) { return arguments.length ? (xSize = _, bhm) : xSize; }\n  /**\n   * Gets / sets the xSpacerSize\n   * (see {@link bubbleHeatmap#xSpacerSize})\n   * @param {number} [_=none]\n   * @returns {bubbleHeatmap | number}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default xSpacerSize = undefined\n   */\n  bhm.xSpacerSize = function(_) { return arguments.length ? (xSpacerSize = _, bhm) : xSpacerSize; }\n  /**\n   * Gets / sets the ySize\n   * (see {@link bubbleHeatmap#ySize})\n   * @param {number} [_=none]\n   * @returns {bubbleHeatmap | number}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default ySize = undefined\n   */\n  bhm.ySize = function(_) { return arguments.length ? (ySize = _, bhm) : ySize; }\n  /**\n   * Gets / sets the ySpacerSize\n   * (see {@link bubbleHeatmap#ySpacerSize})\n   * @param {number} [_=none]\n   * @returns {bubbleHeatmap | number}\n   * @memberof bubbleHeatmap\n   * @property\n   * by default ySpacerSize = undefined\n   */\n  bhm.ySpacerSize = function(_) { return arguments.length ? (ySpacerSize = _, bhm) : ySpacerSize; }\n  // bhm.yKeySortingFunction = function(_) { return arguments.length ? (yKeySortingFunction = _, bhm) : yKeySortingFunction; }\n  // bhm.xKeySortingFunction = function(_) { return arguments.length ? (xKeySortingFunction = _, bhm) : xKeySortingFunction; }\n\n\n\n  function bhm() {\n    var horizontalQ = true; // no orientation in heatmaps. Aids as placeholder in functions that take orient as a parameter\n    var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}\n    var container = setupContainer( selection, namespace, bgcpRect, backgroundFill );\n\n    cellKeys = d3.keys(data);\n    cellKeys.sort(function(a, b){ return xKeySortingFunction(a, b) || yKeySortingFunction(a, b) })\n    log('bubbleHeatmap', 'cells are sorted by', cellKeys)\n\n\n\n    xValues = unique(cellKeys.map(xExtractor));\n    yValues = unique(cellKeys.map(yExtractor));\n    rValues = unique(cellKeys.map(rExtractor));\n    vValues = unique(cellKeys.map(vExtractor));\n    log('bubbleHeatmap', 'x and y keys are', {x: xValues, y:yValues})\n\n\n    var xDim = xValues.length, yDim = yValues.length;\n\n\n    var extent = [Math.min(...rValues) - domainPadding,Math.max(...rValues) + domainPadding];\n\n\n    ySize = calculateWidthOfObject(spaceY, yDim, minObjectSize, maxObjectSize, objectSpacer, overflowQ)\n    xSize = calculateWidthOfObject(spaceX, xDim, minObjectSize, maxObjectSize, objectSpacer, overflowQ)\n    ySpacerSize = calculateWidthOfSpacer(yValues, spaceY, ySize, yDim, objectSpacer, overflowQ)\n    xSpacerSize = calculateWidthOfSpacer(xValues, spaceX, xSize, xDim, objectSpacer, overflowQ)\n    log('bubbleHeatmap', 'size of', {x: xSize, y: ySize})\n\n\n    scale.domain(extent).range([Math.min(minObjectSize/2,   Math.min(ySize, xSize)/2), Math.min(ySize, xSize)/2])\n\n    var ySpacer = groupingSpacer()\n    .horizontalQ(false)\n    .moveby('category').numberOfObjects(yDim)\n    .objectClass(hypenate(objectClass, 'row'))\n    .objectSize(ySize).spacerSize(ySpacerSize)\n    .transitionDuration(transitionDuration).easeFunc(easeFunc)\n    .namespace('row')\n\n    var xSpacer = groupingSpacer()\n    .horizontalQ(true)\n    .moveby('category').numberOfObjects(xDim)\n    .objectClass(objectClass)\n    .objectSize(xSize).spacerSize(xSpacerSize)\n    .transitionDuration(transitionDuration).easeFunc(easeFunc)\n\n\n    ySpacer(container, yValues, 0)\n    container.selectAll('g.'+hypenate(objectClass, 'row'))\n    .each(function(d, i){\n      xSpacer(d3.select(this), xValues, 0)\n    })\n    var cells = container.selectAll('g:not(.to-remove).'+objectClass).data(cellKeys);\n\n    var parentIndexArray = []\n    cells.each(function(d, i){ parentIndexArray.push(Number(d3.select(this).attr('parent-index'))) })\n\n\n    colorFunction = colorFunction.colorBy() == 'index'\n    ? colorFunction.dataExtent([0, Math.max(...parentIndexArray)])\n    : colorFunction.dataExtent([0, Math.max(...vValues)])\n\n    cells.each(function(key, i) {\n      log('bubbleHeatmap', 'each cell', {key: key, index: i, node: d3.select(this).node()})\n\n      var t = d3.select(this),\n      currentData = data[key],\n      value = vExtractor(key, i),\n      radius= rExtractor(key, i),\n      i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'),\n      fillColor = colorFunction(key, value, i, 'fill'), // prevent duplicate computation\n      strokeColor = colorFunction(key, value, i,  'stroke')\n\n      log('bubbleHeatmap', 'radius',{radius: radius, scaled: scale(radius), extent: extent, range:scale.range()})\n\n      var c = safeSelect(t, 'circle', hypenate(objectClass,'circle'))\n      c.attr('cx', xSize / 2)\n      .attr('cy', ySize / 2 )\n      .attr('r', scale(radius))\n      .attr('fill', fillColor)\n      .attr('stroke', strokeColor)\n      .attr('stroke-width', bubbleStrokeWidth)\n\n    })\n\n    tooltip.selection(cells.selectAll('circle.'+hypenate(objectClass, 'circle')))\n    .data(data)\n    // .keys(['r', 'v'])\n    // .header(function(d, i){return hypenate(data[d][xKey], data[d][yKey]) })\n\n    tooltip()\n\n\n  }\n\n  return bhm;\n}\n\nexport {bubbleHeatmap}\n","import {hypenate, safeSelect} from './helpers';\nimport {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer, log} from './utils';\nimport {unique} from './array-functions';\nimport {groupingSpacer} from './grouping-spacer';\nimport {colorFunction as CF} from './color-function';\nimport {tooltip as TTip} from './tooltip';\n\n\n/**\n * Creates a heatmap\n *\n * {@link https://sumneuron.gitlab.io/d3sm/demos/heatmap-heatmap/index.html Demo}\n * @constructor heatmap\n * @param {d3.selection} selection\n * @namespace heatmap\n * @returns {function} heatmap\n */\nfunction heatmap( selection ) {\n  var\n  /**\n  * Data to plot. Assumed to be a object, where each key corresponds to a cell\n  * (see {@link heatmap#data})\n  * @param {Object} [data=undefined]\n  * @memberof heatmap#\n  * @property\n  */\n  data,\n\n  /**\n  * Amount of horizontal space (in pixels) avaible to render the heatmap in\n  * (see {@link heatmap#spaceX})\n  * @param {number} [spaceX=undefined]\n  * @memberof heatmap#\n  * @property\n  */\n  spaceX,\n  /**\n  * Amount of vertical space (in pixels) avaible to render the heatmap in\n  * (see {@link heatmap.spaceY})\n  * @param {number} [spaceY=undefined]\n  * @memberof heatmap#\n  * @property\n  */\n  spaceY,\n\n  /**\n  * The internal key of the cell specifiying to which x axis key it belongs\n  * (see {@link heatmap.xKey})\n  * @param {string} [xKey='x']\n  * @memberof heatmap#\n  * @property\n  */\n  xKey = 'x',\n  /**\n  * The internal key of the cell specifiying to which y axis key it belongs\n  * (see {@link heatmap.yKey})\n  * @param {string} [yKey='y']\n  * @memberof heatmap#\n  * @property\n  */\n  yKey = 'y',\n\n  /**\n  * The internal key of the cell specifiying what value to use to determine the color\n  * (see {@link heatmap.vKey})\n  * @param {string} [vKey='v']\n  * @memberof heatmap#\n  * @property\n  */\n  vKey = 'v',\n\n  /**\n  * Function for extracting the the value from xKey.\n  * (see {@link heatmap.xExtractor})\n  * @param {function} [xExtractor=function(key, i) { return data[key][xKey] }]\n  * @returns {string}\n  * @memberof heatmap#\n  * @property\n  */\n  xExtractor = function(key, i) {return data[key][xKey] },\n  /**\n  * Function for extracting the the value from yKey.\n  * (see {@link heatmap.yExtractor})\n  * @param {function} [yExtractor=function(key, i) { return data[key][yKey] }]\n  * @returns {string}\n  * @memberof heatmap#\n  * @property\n  */\n  yExtractor = function(key, i) { return data[key][yKey] },\n\n  /**\n  * Function for extracting the the value from vKey.\n  * (see {@link heatmap.vExtractor})\n  * @param {function} [vExtractor=function(key, i) { return data[key][vKey] }]\n  * @returns {number}\n  * @memberof heatmap#\n  * @property\n  */\n  vExtractor = function(key, i) { return data[key][vKey] },\n\n\n  /**\n  * Whether or not to allow heatmap to render elements pass the main spatial dimension\n  * given the orientation (see {@link heatmap#orient}), where {@link heatmap#orient}=\"bottom\" or {@link heatmap#orient}=\"top\"\n  * the main dimension is {@link heatmap#spaceX} and where {@link heatmap#orient}=\"left\" or {@link heatmap#orient}=\"right\"\n  * the main dimension is {@link heatmap#spaceY}\n  * @param {boolean} [overflowQ=false]\n  * @memberof heatmap#\n  * @property\n  */\n  overflowQ = false,\n\n  /**\n  * Default space for the spacer (percentage) of main dimension given the orientation\n  * (see {@link heatmap#orient}), where {@link heatmap#orient}=\"horizontal\"\n  * the main dimension is {@link heatmap#spaceX} and where {@link heatmap#orient}=\"vertical\"\n  * the main dimension is {@link heatmap#spaceY} between bubbles\n  * @param {number} [objectSpacer=0.0]\n  * @memberof heatmap#\n  * @property\n  */\n  objectSpacer = 0.0,\n  /**\n  * The minimum size that an object can be in the y dimension\n  * @param {number} [minObjectSize=50]\n  * @memberof heatmap#\n  * @property\n  */\n  yMinObjectSize = 50,\n  /**\n  * The minimum size that an object can be in the x dimension\n  * @param {number} [minObjectSize=50]\n  * @memberof heatmap#\n  * @property\n  */\n  xMinObjectSize = 50,\n  /**\n  * The maximum size that an object can be in the x dimension\n  * @param {number} [maxObjectSize=100]\n  * @memberof heatmap#\n  * @property\n  */\n  xMaxObjectSize = 100,\n  /**\n  * The maximum size that an object can be in the y dimension\n  * @param {number} [maxObjectSize=100]\n  * @memberof heatmap#\n  * @property\n  */\n  yMaxObjectSize = 100,\n\n\n  /**\n  * The stroke width of the bubbles\n  * @param {number} [objectStrokeWidth=2]\n  * @memberof heatmap#\n  * @property\n  */\n  objectStrokeWidth = 2,\n  // colorFunc = colorFunction(),\n\n  /**\n  * Color of the background\n  * @param {string} [backgroundFill=\"transparent\"]\n  * @memberof heatmap#\n  * @property\n  */\n  backgroundFill = 'transparent',\n  /**\n  * Namespace for all items made by this instance of heatmap\n  * @param {string} [namespace=\"d3sm-heatmap\"]\n  * @memberof heatmap#\n  * @property\n  */\n  namespace = 'd3sm-heatmap',\n  /**\n  * Class name for heatmap container (<g> element)\n  * @param {string} [objectClass=\"heatmap\"]\n  * @memberof heatmap#\n  * @property\n  */\n  objectClass = 'heatmap',\n  /**\n  * Duration of all transitions of this element\n  * @param {number} [transitionDuration=1000]\n  * @memberof heatmap#\n  * @property\n  */\n  transitionDuration = 1000,\n  /**\n  * Easing function for transitions\n  * @param {d3.ease} [easeFunc=d3.easeExp]\n  * @memberof heatmap#\n  * @property\n  */\n  easeFunc = d3.easeExp,\n\n  /**\n  * Stores the keys of all the cells\n  * Calculated after heatmap called.\n  * @param {string[]} [cellKeys=undefined]\n  * @memberof heatmap#\n  * @property\n  */\n  cellKeys,\n  /**\n  * Stores the list of unique xValues\n  * Calculated after heatmap called.\n  * @param {string[]} [xValues=undefined]\n  * @memberof heatmap#\n  * @property\n  */\n  xValues,\n  /**\n  * Stores the list of unique yValues\n  * Calculated after heatmap called.\n  * @param {string[]} [yValues=undefined]\n  * @memberof heatmap#\n  * @property\n  */\n  yValues,\n  /**\n  * Stores the list of unique vValues\n  * Calculated after heatmap called.\n  * @param {string[]} [vValues=undefined]\n  * @memberof heatmap#\n  * @property\n  */\n  vValues,\n\n  xKeySortingFunction = function(a, b) { return xValues.indexOf(xExtractor(a)) - xValues.indexOf(xExtractor(b)) },\n  yKeySortingFunction = function(a, b) { return yValues.indexOf(yExtractor(a)) - yValues.indexOf(yExtractor(b)) },\n  vKeySortingFunction = function(a, b) { return vValues.indexOf(vExtractor(a)) - yValues.indexOf(vExtractor(b)) },\n\n  /**\n  * Instance of ColorFunction with .colorBy set to 'category'\n  * @function colorFunction\n  * @memberof heatmap#\n  * @property\n  */\n  colorFunction = CF().colorBy('category'),\n  /**\n  * Instance of Tooltip\n  * @function tooltip\n  * @memberof heatmap#\n  * @property\n  */\n  tooltip = TTip(),\n\n  /**\n  * store the size the heatmap could be in the x dimension\n  * the actuall size of the heatmap will be the min of xSize and {@link heatmap#ySize}\n  * Calculated after heatmap called.\n  * @param {string[]} [xSize=undefined]\n  * @memberof heatmap#\n  * @property\n  */\n  xSize,\n  /**\n  * store the size of the spacer in the x dimension\n  * Calculated after heatmap called.\n  * @param {string[]} [xSpacerSize=undefined]\n  * @memberof heatmap#\n  * @property\n  */\n  xSpacerSize,\n\n  /**\n  * store the size the heatmap could be in the y dimension\n  * the actuall size of the heatmap will be the min of xSize and {@link heatmap#xSize}\n  * Calculated after heatmap called.\n  * @param {string[]} [ySize=undefined]\n  * @memberof heatmap#\n  * @property\n  */\n  ySize,\n  /**\n  * store the size of the spacer in the y dimension.\n  * Calculated after heatmap called.\n  * @param {string[]} [xSpacerSize=undefined]\n  * @memberof heatmap#\n  * @property\n  */\n  ySpacerSize\n\n  /**\n   * Gets or sets the selection in which items are manipulated\n   * @param {d3.selection} [_=none]\n   * @returns {heatmap | d3.selection}\n   * @memberof heatmap\n   * @property\n   * by default selection = selection\n   */\n  hm.selection = function(_) { return arguments.length ? (selection = _, hm) : selection; }\n  /**\n   * Gets or sets the data\n   * (see {@link heatmap#data})\n   * @param {number} [_=none]\n   * @returns {heatmap | object}\n   * @memberof heatmap\n   * @property\n   */\n  hm.data = function(_) { return arguments.length ? (data = _, hm) : data; }\n  // hm.orient = function(_) { return arguments.length ? (orient = _, hm) : orient; }\n  /**\n   * Gets or sets the amount of horizontal space in which items are manipulated\n   * (see {@link heatmap#spaceX})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {heatmap | number}\n   * @memberof heatmap\n   * @property\n   * by default spaceX = undefined\n   */\n  hm.spaceX = function(_) { return arguments.length ? (spaceX = _, hm) : spaceX; }\n  /**\n   * Gets or sets the amount of vertical space in which items are manipulated\n   * (see {@link heatmap#spaceY})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {heatmap | number}\n   * @memberof heatmap\n   * @property\n   * by default spaceY = undefined\n   */\n  hm.spaceY = function(_) { return arguments.length ? (spaceY = _, hm) : spaceY; }\n\n  /**\n   * Gets or sets the xKey\n   * (see {@link heatmap#xKey})\n   * @param {string} [_=none]\n   * @returns {heatmap | string}\n   * @memberof heatmap\n   * @property\n   * by default xKey = 'x'\n   */\n  hm.xKey = function(_) { return arguments.length ? (xKey = _, hm) : xKey; }\n  /**\n   * Gets or sets the yKey\n   * (see {@link heatmap#yKey})\n   * @param {string} [_=none]\n   * @returns {heatmap | string}\n   * @memberof heatmap\n   * @property\n   * by default yKey = 'y'\n   */\n  hm.yKey = function(_) { return arguments.length ? (yKey = _, hm) : yKey; }\n\n  /**\n   * Gets or sets the vKey\n   * (see {@link heatmap#vKey})\n   * @param {string} [_=none]\n   * @returns {heatmap | string}\n   * @memberof heatmap\n   * @property\n   * by default vKey = 'y'\n   */\n  hm.vKey = function(_) { return arguments.length ? (vKey = _, hm) : vKey; }\n\n  /**\n   * Gets or sets the cellKeys\n   * (see {@link heatmap#cellKeys})\n   * @param {string[]} [_=none]\n   * @returns {heatmap | string[]}\n   * @memberof heatmap\n   * @property\n   * by default cellKeys = undefined\n   */\n  hm.cellKeys = function(_) { return arguments.length ? (cellKeys = _, hm) : cellKeys; }\n  /**\n   * Gets or sets the xValues\n   * (see {@link heatmap#xValues})\n   * @param {string[]} [_=none]\n   * @returns {heatmap | string[]}\n   * @memberof heatmap\n   * @property\n   * by default xValues = undefined\n   */\n  hm.xValues = function(_) { return arguments.length ? (xValues = _, hm) : xValues; }\n  /**\n   * Gets or sets the yValues\n   * (see {@link heatmap#yValues})\n   * @param {string[]} [_=none]\n   * @returns {heatmap | string[]}\n   * @memberof heatmap\n   * @property\n   * by default yValues = undefined\n   */\n  hm.yValues = function(_) { return arguments.length ? (yValues = _, hm) : yValues; }\n  /**\n   * Gets or sets the vValues\n   * (see {@link heatmap#vValues})\n   * @param {number[]} [_=none]\n   * @returns {heatmap | number[]}\n   * @memberof heatmap\n   * @property\n   * by default vValues = undefined\n   */\n  hm.vValues = function(_) { return arguments.length ? (vValues = _, hm) : vValues; }\n\n\n  /**\n   * Gets or sets the xExtractor\n   * (see {@link heatmap#xExtractor})\n   * @param {function} [_=none]\n   * @returns {heatmap | function}\n   * @memberof heatmap\n   * @property\n   * by default xExtractor = undefined\n   */\n  hm.xExtractor = function(_) { return arguments.length ? (xExtractor = _, hm) : xExtractor; }\n  /**\n   * Gets or sets the yExtractor\n   * (see {@link heatmap#yExtractor})\n   * @param {function} [_=none]\n   * @returns {heatmap | function}\n   * @memberof heatmap\n   * @property\n   * by default yExtractor = undefined\n   */\n  hm.yExtractor = function(_) { return arguments.length ? (yExtractor = _, hm) : yExtractor; }\n  /**\n   * Gets or sets the vExtractor\n   * (see {@link heatmap#vExtractor})\n   * @param {function} [_=none]\n   * @returns {heatmap | function}\n   * @memberof heatmap\n   * @property\n   * by default vExtractor = undefined\n   */\n  hm.vExtractor = function(_) { return arguments.length ? (vExtractor = _, hm) : vExtractor; }\n\n  /**\n   * Gets / sets whether or not heatmap is allowed to go beyond specified dimensions\n   * (see {@link heatmap#spaceX})\n   * @param {boolean} [_=none]\n   * @returns {heatmap | boolean}\n   * @memberof heatmap\n   * @property\n   * by default overflowQ = false\n   */\n  hm.overflowQ = function(_) { return arguments.length ? (overflowQ = _, hm) : overflowQ; }\n  /**\n   * Gets / sets objectSpacer\n   * (see {@link heatmap#objectSpacer})\n   * @param {number} [_=none]\n   * @returns {heatmap | number}\n   * @memberof heatmap\n   * @property\n   * by default objectSpacer = 0.0\n   */\n  hm.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, objectSpacer) : data; }\n  /**\n   * Gets / sets the minObjectSize\n   * (see {@link heatmap#minObjectSize})\n   * @param {number} [_=none]\n   * @returns {heatmap | number}\n   * @memberof heatmap\n   * @property\n   * by default minObjectSize = 50\n   */\n  hm.yMinObjectSize = function(_) { return arguments.length ? (yMinObjectSize = _, hm) : yMinObjectSize; }\n  /**\n   * Gets / sets the maxObjectSize\n   * (see {@link heatmap#maxObjectSize})\n   * @param {number} [_=none]\n   * @returns {heatmap | number}\n   * @memberof heatmap\n   * @property\n   * by default maxObjectSize = 100\n   */\n  hm.yMaxObjectSize = function(_) { return arguments.length ? (yMaxObjectSize = _, hm) : yMaxObjectSize; }\n  /**\n   * Gets / sets the minObjectSize\n   * (see {@link heatmap#minObjectSize})\n   * @param {number} [_=none]\n   * @returns {heatmap | number}\n   * @memberof heatmap\n   * @property\n   * by default minObjectSize = 50\n   */\n  hm.xMinObjectSize = function(_) { return arguments.length ? (xMinObjectSize = _, hm) : xMinObjectSize; }\n  /**\n   * Gets / sets the maxObjectSize\n   * (see {@link heatmap#maxObjectSize})\n   * @param {number} [_=none]\n   * @returns {heatmap | number}\n   * @memberof heatmap\n   * @property\n   * by default maxObjectSize = 100\n   */\n  hm.xMaxObjectSize = function(_) { return arguments.length ? (xMaxObjectSize = _, hm) : xMaxObjectSize; }\n  /**\n   * Gets / sets the objectStrokeWidth\n   * (see {@link heatmap#objectStrokeWidth})\n   * @param {number} [_=none]\n   * @returns {heatmap | number}\n   * @memberof heatmap\n   * @property\n   * by default objectStrokeWidth = 2\n   */\n  hm.objectStrokeWidth = function(_) { return arguments.length ? (objectStrokeWidth = _, hm) : objectStrokeWidth; }\n  /**\n   * Gets / sets the backgroundFill\n   * (see {@link heatmap#backgroundFill})\n   * @param {string} [_=none]\n   * @returns {heatmap | string}\n   * @memberof heatmap\n   * @property\n   * by default backgroundFill = 'transparent'\n   */\n  hm.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, hm) : backgroundFill; }\n  /**\n   * Gets / sets the namespace\n   * (see {@link heatmap#namespace})\n   * @param {string} [_=none]\n   * @returns {heatmap | string}\n   * @memberof heatmap\n   * @property\n   * by default namespace = 'd3sm-heatmap'\n   */\n  hm.namespace = function(_) { return arguments.length ? (namespace = _, hm) : namespace; }\n  /**\n   * Gets / sets the objectClass\n   * (see {@link heatmap#objectClass})\n   * @param {string} [_=none]\n   * @returns {heatmap | string}\n   * @memberof heatmap\n   * @property\n   * by default objectClass = 'tick-group'\n   */\n  hm.objectClass = function(_) { return arguments.length ? (objectClass = _, hm) : objectClass; }\n  /**\n   * Gets / sets the transitionDuration\n   * (see {@link heatmap#transitionDuration})\n   * @param {number} [_=none]\n   * @returns {heatmap | number}\n   * @memberof heatmap\n   * @property\n   * by default transitionDuration = 1000\n   */\n  hm.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, hm) : transitionDuration; }\n  /**\n   * Gets / sets the easeFunc\n   * (see {@link heatmap#easeFunc})\n   * @param {d3.ease} [_=none]\n   * @returns {heatmap | d3.ease}\n   * @memberof heatmap\n   * @property\n   * by default easeFunc = d3.easeExp\n   */\n  hm.easeFunc = function(_) { return arguments.length ? (easeFunc = _, hm) : easeFunc; }\n\n  /**\n   * Gets / sets the tooltip\n   * (see {@link heatmap#tooltip})\n   * @param {tooltip} [_=none]\n   * @returns {heatmap | tooltip}\n   * @memberof heatmap\n   * @property\n   * by default tooltip = tooltip()\n   */\n  hm.tooltip = function(_) { return arguments.length ? (tooltip = _, hm) : tooltip; }\n\n  /**\n   * Gets / sets the colorFunction\n   * (see {@link heatmap#colorFunction})\n   * @param {colorFunction} [_=none]\n   * @returns {heatmap | colorFunction}\n   * @memberof heatmap\n   * @property\n   * by default colorFunction = colorFunction()\n   */\n  hm.colorFunction = function(_) { return arguments.length ? (colorFunction = _, hm) : colorFunction; }\n\n  /**\n   * Gets / sets the xSize\n   * (see {@link heatmap#xSize})\n   * @param {number} [_=none]\n   * @returns {heatmap | number}\n   * @memberof heatmap\n   * @property\n   * by default xSize = undefined\n   */\n  hm.xSize = function(_) { return arguments.length ? (xSize = _, hm) : xSize; }\n  /**\n   * Gets / sets the xSpacerSize\n   * (see {@link heatmap#xSpacerSize})\n   * @param {number} [_=none]\n   * @returns {heatmap | number}\n   * @memberof heatmap\n   * @property\n   * by default xSpacerSize = undefined\n   */\n  hm.xSpacerSize = function(_) { return arguments.length ? (xSpacerSize = _, hm) : xSpacerSize; }\n  /**\n   * Gets / sets the ySize\n   * (see {@link heatmap#ySize})\n   * @param {number} [_=none]\n   * @returns {heatmap | number}\n   * @memberof heatmap\n   * @property\n   * by default ySize = undefined\n   */\n  hm.ySize = function(_) { return arguments.length ? (ySize = _, hm) : ySize; }\n  /**\n   * Gets / sets the ySpacerSize\n   * (see {@link heatmap#ySpacerSize})\n   * @param {number} [_=none]\n   * @returns {heatmap | number}\n   * @memberof heatmap\n   * @property\n   * by default ySpacerSize = undefined\n   */\n  hm.ySpacerSize = function(_) { return arguments.length ? (ySpacerSize = _, hm) : ySpacerSize; }\n  // hm.yKeySortingFunction = function(_) { return arguments.length ? (yKeySortingFunction = _, hm) : yKeySortingFunction; }\n  // hm.xKeySortingFunction = function(_) { return arguments.length ? (xKeySortingFunction = _, hm) : xKeySortingFunction; }\n\n\n\n  function hm() {\n    var horizontalQ = true; // no orientation in heatmaps. Aids as placeholder in functions that take orient as a parameter\n    var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}\n    var container = setupContainer( selection, namespace, bgcpRect, backgroundFill );\n\n    cellKeys = d3.keys(data);\n\n    xValues = unique(cellKeys.map(xExtractor));\n    yValues = unique(cellKeys.map(yExtractor));\n    vValues = unique(cellKeys.map(vExtractor));\n\n    cellKeys.sort(function(a, b){ return xKeySortingFunction(a, b) || yKeySortingFunction(a, b) })\n    log('heatmap', 'cells are sorted by', cellKeys)\n\n\n\n    log('heatmap', 'x and y keys are', {x: xValues, y:yValues})\n\n\n    var xDim = xValues.length, yDim = yValues.length;\n\n\n    ySize = calculateWidthOfObject(spaceY, yDim, yMinObjectSize, yMaxObjectSize, objectSpacer, overflowQ)\n    xSize = calculateWidthOfObject(spaceX, xDim, xMinObjectSize, xMaxObjectSize, objectSpacer, overflowQ)\n    ySpacerSize = calculateWidthOfSpacer(yValues, spaceY, ySize, yDim, objectSpacer, overflowQ)\n    xSpacerSize = calculateWidthOfSpacer(xValues, spaceX, xSize, xDim, objectSpacer, overflowQ)\n    // console.table({\n    //     x:{\n    //       object: xSize,\n    //       spacer: xSpacerSize,\n    //       dim: xDim\n    //     },\n    //     y:{\n    //       object: ySize,\n    //       spacer: ySpacerSize,\n    //       dim: yDim\n    //     }\n    //\n    // })\n    log('heatmap', 'size of', {x: xSize, y: ySize})\n\n\n\n    var ySpacer = groupingSpacer()\n    .horizontalQ(false)\n    .moveby('category')\n    .numberOfObjects(yDim)\n    .objectClass(hypenate(objectClass, 'row'))\n    .objectSize(ySize + ySpacerSize)\n    .spacerSize(0)\n    .transitionDuration(transitionDuration)\n    .easeFunc(easeFunc)\n    .namespace('row')\n\n    var xSpacer = groupingSpacer()\n    .horizontalQ(true)\n    .moveby('category')\n    .numberOfObjects(xDim)\n    .objectClass(objectClass)\n    .objectSize(xSize + xSpacerSize)\n    .spacerSize(0)\n    .transitionDuration(transitionDuration)\n    .easeFunc(easeFunc)\n\n\n    ySpacer(container, yValues, 0)\n    container.selectAll('g.'+hypenate(objectClass, 'row'))\n    .each(function(d, i){ xSpacer(d3.select(this), xValues, 0) })\n\n    var cells = container.selectAll('g:not(.to-remove).'+objectClass)\n\n\n    if (cellKeys.length != yValues.length * xValues.length) {\n      var lookup = {}\n      cellKeys.map(function(k, i){\n        lookup[xExtractor(k)+'::'+yExtractor(k)] = k\n      })\n\n      var positionedCellKeys = []\n      for (var i = 0; i < yValues.length; i++) {\n        for (var j = 0; j < xValues.length; j++) {\n          var lookupValue = lookup[xValues[j]+\"::\"+yValues[i]]\n          if (lookupValue == undefined) {\n            positionedCellKeys.push(undefined)\n          } else {\n            positionedCellKeys.push(lookupValue)\n          }\n        }\n      }\n\n      cells.data(positionedCellKeys);\n\n      // maybe breaks this\n      // !!!!!! IMPORTANT NOTE TODO LOOK HERE BUG\n      cellKeys = positionedCellKeys\n    } else {\n      cells.data(cellKeys);\n    }\n\n\n\n    var parentIndexArray = []\n    cells.each(function(d, i){ parentIndexArray.push(Number(d3.select(this).attr('parent-index'))) })\n\n    colorFunction = colorFunction.colorBy() == 'index'\n    ? colorFunction.dataExtent([0, Math.max(...parentIndexArray)])\n    : colorFunction.dataExtent([0, Math.max(...vValues)])\n\n    cells.each(function(key, i) {\n      log('heatmap', 'each cell', {key: key, index: i, node: d3.select(this).node()})\n\n      var t = d3.select(this)\n      if (key == undefined) {return}\n      var currentData = data[key],\n      value = vExtractor(key, i),\n      i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'),\n      fillColor = colorFunction(key, value, i, 'fill'), // prevent duplicate computation\n      strokeColor = colorFunction(key, value, i,  'stroke')\n\n      var c = safeSelect(t, 'rect', hypenate(objectClass,'rect'))\n      c.attr('width', xSize + xSpacerSize - objectStrokeWidth)\n      .attr('height', ySize + ySpacerSize - objectStrokeWidth)\n      .attr('fill', fillColor)\n      .attr('x', objectStrokeWidth/2)\n      .attr('y', objectStrokeWidth/2)\n      .attr('stroke', \"#000\")\n      .attr('stroke-width', objectStrokeWidth)\n\n    })\n\n    tooltip.selection(cells.selectAll('rect.'+hypenate(objectClass, 'rect')))\n    .data(data)\n    // .keys(['r', 'v'])\n    // .header(function(d, i){return hypenate(data[d][xKey], data[d][yKey]) })\n\n    tooltip()\n\n\n  }\n\n  return hm;\n}\n\nexport {heatmap}\n","import {hypenate, safeSelect} from './helpers';\nimport {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer, whiskerPath} from './utils';\nimport {unique, hasQ, flatten} from './array-functions';\nimport {groupingSpacer} from './grouping-spacer';\nimport {colorFunction as CF} from './color-function';\nimport {tooltip as TTip} from './tooltip';\n/*******************************************************************************\n**                                                                            **\n**                                                                            **\n**                             BOX AND WHISKER                                **\n**                                                                            **\n**                                                                            **\n*******************************************************************************/\n/**\n * Creates a boxwhisker\n *\n * {@link https://sumneuron.gitlab.io/d3sm/demos/box-whiskers/index.html Demo}\n * @constructor boxwhisker\n * @param {d3.selection} selection\n * @namespace boxwhisker\n * @returns {function} boxwhisker\n */\nexport function boxwhisker( selection ) {\n  var\n  /**\n  * Data to plot. Assumed to be a object, where each key corresponds to a box\n  * (see {@link boxwhisker#data})\n  * @param {Object} [data=undefined]\n  * @memberof boxwhisker#\n  * @property\n  */\n  data,\n  /**\n  * Which direction to render the boxes in\n  * (see {@link boxwhisker#orient})\n  * @param {number} [orient='horizontal']\n  * @memberof boxwhisker#\n  * @property\n  */\n  orient = 'horizontal',\n  /**\n  * Amount of horizontal space (in pixels) avaible to render the boxwhisker in\n  * (see {@link boxwhisker#spaceX})\n  * @param {number} [spaceX=undefined]\n  * @memberof boxwhisker#\n  * @property\n  */\n  spaceX,\n  /**\n  * Amount of vertical space (in pixels) avaible to render the boxwhisker in\n  * (see {@link boxwhisker.spaceY})\n  * @param {number} [spaceY=undefined]\n  * @memberof boxwhisker#\n  * @property\n  */\n  spaceY,\n  /**\n  * Whether or not to allow boxwhisker to render elements pass the main spatial dimension\n  * given the orientation (see {@link boxwhisker#orient}), where {@link boxwhisker#orient}=\"horizontal\"\n  * the main dimension is {@link boxwhisker#spaceX} and where {@link boxwhisker#orient}=\"vertical\"\n  * the main dimension is {@link boxwhisker#spaceY}\n  * @param {boolean} [overflowQ=false]\n  * @memberof boxwhisker#\n  * @property\n  */\n  overflowQ = true,\n\n  /**\n  * An array - putatively of other arrays - depicting how boxes should be arranged\n  * @param {Array[]} [grouping=undefined]\n  * @memberof boxwhisker#\n  * @property\n  */\n  grouping,\n  quartilesKey = 'quartiles', // key in object where quartiles are stored\n  quartilesKeys = [\"0.00\", \"0.25\", \"0.50\", \"0.75\", \"1.00\"], // keys in quartiles object mapping values to min, q1, q2, q3 and max\n\n\n  /**\n  * How to get the value of the boxwhisker\n  * @param {function} [valueExtractor=function(key, index) { return data[key][quartilesKey] }]\n  * @memberof boxwhisker#\n  * @property\n  */\n  valueExtractor = function(key, index) { return data[key][quartilesKey] },\n  /**\n  * How to sort the boxes - if {@link boxwhisker#grouping} is not provided.\n  * @param {function} [sortingFunction=descending]\n  * @memberof boxwhisker#\n  * @property\n  * default\n  * function(keyA, keyB) {return d3.descending(\n  *   valueExtractor(keyA)[quartilesKeys[4]],\n  *   valueExtractor(keyB)[quartilesKeys[4]]\n  * )}\n  */\n  sortingFunction = function(keyA, keyB) {return d3.descending(\n    valueExtractor(keyA)[quartilesKeys[4]],\n    valueExtractor(keyB)[quartilesKeys[4]]\n  )},\n  /**\n  * The scale for which boxwhisker values should be transformed by\n  * @param {d3.scale} [scale=d3.scaleLinear]\n  * @memberof boxwhisker#\n  * @property\n  */\n  scale = d3.scaleLinear(),\n  /**\n  * The padding for the domain of the scale (see {@link boxwhisker#scale})\n  * @param {number} [domainPadding=0.5]\n  * @memberof boxwhisker#\n  * @property\n  */\n  domainPadding = 0.5,\n  /**\n  * Default space for the spacer (percentage) of main dimension given the orientation\n  * (see {@link boxwhisker#orient}), where {@link boxwhisker#orient}=\"horizontal\"\n  * the main dimension is {@link boxwhisker#spaceX} and where {@link boxwhisker#orient}=\"vertical\"\n  * the main dimension is {@link boxwhisker#spaceY} between boxes\n  * @param {number} [objectSpacer=0.05]\n  * @memberof boxwhisker#\n  * @property\n  */\n  objectSpacer = 0.05,\n  /**\n  * The minimum size that an object can be\n  * @param {number} [minObjectSize=15]\n  * @memberof boxwhisker#\n  * @property\n  */\n  minObjectSize = 15,\n  /**\n  * The maximum size that an object can be\n  * @param {number} [maxObjectSize=50]\n  * @memberof boxwhisker#\n  * @property\n  */\n  maxObjectSize = 50,\n  /**\n  * Percent of box and whisker size that whiskers will be rendered\n  * see {@link boxwhisker#objectSize}\n  * @param {number} [whiskerWidthPercent=0.6]\n  * @memberof boxwhisker#\n  * @property\n  */\n  whiskerWidthPercent = .6,\n  /**\n  * Instance of ColorFunction\n  * @param {function} [colorFunction = colorFunction()]\n  * @memberof boxwhisker#\n  * @property\n  */\n  colorFunction = CF(),\n  /**\n  * The stroke width of the boxes\n  * @param {number} [boxStrokeWidth=2]\n  * @memberof boxwhisker#\n  * @property\n  */\n  boxStrokeWidth = 2,\n  /**\n  * The stroke width of the whiskers\n  * @param {number} [whiskerStrokeWidth=2]\n  * @memberof boxwhisker#\n  * @property\n  */\n  whiskerStrokeWidth = 2,\n\n  /**\n  * Color of the background\n  * @param {string} [backgroundFill=\"transparent\"]\n  * @memberof boxwhisker#\n  * @property\n  */\n  backgroundFill = 'transparent',\n  /**\n  * Namespace for all items made by this instance of boxwhisker\n  * @param {string} [namespace=\"d3sm-box-whisker\"]\n  * @memberof boxwhisker#\n  * @property\n  */\n  namespace = 'd3sm-box-whisker',\n  /**\n  * Class name for boxwhisker container (<g> element)\n  * @param {string} [objectClass=\"box-whisk\"]\n  * @memberof boxwhisker#\n  * @property\n  */\n  objectClass = 'box-whisk',\n\n  /**\n  * Duration of all transitions of this element\n  * @param {number} [transitionDuration=1000]\n  * @memberof boxwhisker#\n  * @property\n  */\n  transitionDuration = 1000,\n  /**\n  * Easing function for transitions\n  * @param {d3.ease} [easeFunc=d3.easeExp]\n  * @memberof boxwhisker#\n  * @property\n  */\n  easeFunc = d3.easeExp,\n\n  /**\n  * The keys of the boxes\n  * @param {string[]} [boxKeys=undefined]\n  * @memberof boxwhisker#\n  * @property\n  */\n  boxKeys,\n  /**\n  * The values of the boxes\n  * @param {string[]} [boxValues=undefined]\n  * @memberof boxwhisker#\n  * @property\n  */\n  boxValues,\n  /**\n  * The objectSize (actual width) used by the boxes\n  * @param {number} [objectSize=undefined]\n  * @memberof boxwhisker#\n  * @property\n  */\n  objectSize,\n  /**\n  * The spacerSize (actual width) used by the spacers between the boxes\n  * @param {number} [spacerSize=undefined]\n  * @memberof boxwhisker#\n  * @property\n  */\n  spacerSize,\n  /**\n  * Instance of Tooltip\n  * @param {function} [tooltip=tooltip()]\n  * @memberof boxwhisker#\n  * @property\n  */\n  tooltip = TTip()\n  /**\n   * Gets or sets the selection in which items are manipulated\n   * @param {d3.selection} [_=none]\n   * @returns {boxwhisker | d3.selection}\n   * @memberof boxwhisker\n   * @property\n   * by default selection = selection\n   */\n  boxwhisker.selection = function(_) { return arguments.length ? (selection = _, boxwhisker) : selection; };\n  /**\n   * Gets or sets the data\n   * (see {@link boxwhisker#data})\n   * @param {number} [_=none]\n   * @returns {boxwhisker | object}\n   * @memberof boxwhisker\n   * @property\n   */\n  boxwhisker.data = function(_) { return arguments.length ? (data = _, boxwhisker) : data; };\n  /**\n   * Gets or sets the orient of the boxes\n   * (see {@link boxwhisker#orient})\n   * @param {number} [_=none]\n   * @returns {boxwhisker | object}\n   * @memberof boxwhisker\n   * @property\n   */\n  boxwhisker.orient = function(_) { return arguments.length ? (orient = _, boxwhisker) : orient; };\n  /**\n   * Gets or sets the amount of horizontal space in which items are manipulated\n   * (see {@link boxwhisker#spaceX})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {boxwhisker | number}\n   * @memberof boxwhisker\n   * @property\n   * by default spaceX = undefined\n   */\n  boxwhisker.spaceX = function(_) { return arguments.length ? (spaceX = _, boxwhisker) : spaceX; };\n  /**\n   * Gets or sets the amount of vertical space in which items are manipulated\n   * (see {@link boxwhisker#spaceY})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {boxwhisker | number}\n   * @memberof boxwhisker\n   * @property\n   * by default spaceY = undefined\n   */\n  boxwhisker.spaceY = function(_) { return arguments.length ? (spaceY = _, boxwhisker) : spaceY; };\n  /**\n   * Gets / sets whether or not boxwhisker is allowed to go beyond specified dimensions\n   * (see {@link boxwhisker#overflowQ})\n   * @param {boolean} [_=none]\n   * @returns {boxwhisker | boolean}\n   * @memberof boxwhisker\n   * @property\n   * by default overflowQ = false\n   */\n  boxwhisker.overflowQ = function(_) { return arguments.length ? (overflowQ = _, boxwhisker) : overflowQ; };\n  /**\n   * Gets / sets the grouping of the boxes\n   * (see {@link boxwhisker#grouping})\n   * @param {Array[]} [_=none]\n   * @returns {boxwhisker | Array[]}\n   * @memberof boxwhisker\n   * @property\n   * by default grouping = undefined\n   */\n  boxwhisker.grouping = function(_) { return arguments.length ? (grouping = _, boxwhisker) : grouping; };\n  /**\n   * Gets / sets the quartilesKey\n   * (see {@link boxwhisker#quartilesKey})\n   * @param {string} [_=none]\n   * @returns {boxwhisker | string}\n   * @memberof boxwhisker\n   * @property\n   * by default quartilesKey = quartiles\n   */\n  boxwhisker.quartilesKey = function(_) { return arguments.length ? (quartilesKey = _, boxwhisker) : quartilesKey; };\n  /**\n   * Gets / sets the quartilesKeys\n   * (see {@link boxwhisker#quartilesKeys})\n   * @param {string[]} [_=none]\n   * @returns {boxwhisker | string[]}\n   * @memberof boxwhisker\n   * @property\n   * by default quartilesKeys = [Q0, Q1, Q2, Q3, Q4]\n   */\n  boxwhisker.quartilesKeys = function(_) { return arguments.length ? (quartilesKeys = _, boxwhisker) : quartilesKeys; };\n  /**\n   * Gets / sets the valueExtractor\n   * (see {@link boxwhisker#valueExtractor})\n   * @param {function} [_=none]\n   * @returns {boxwhisker | function}\n   * @memberof boxwhisker\n   * @property\n   * by default valueExtractor = function(key, index) { return data[key][quartilesKey] },\n   */\n\n  boxwhisker.valueExtractor = function(_) { return arguments.length ? (valueExtractor = _, boxwhisker) : valueExtractor; };\n  /**\n   * Gets / sets the sortingFunction\n   * (see {@link boxwhisker#sortingFunction})\n   * @param {function} [_=none]\n   * @returns {boxwhisker | function}\n   * @memberof boxwhisker\n   * @property\n   * by default sortingFunction = function(keyA, keyB) {return d3.descending(\n   *   valueExtractor(keyA)[quartilesKeys[4]],\n   *   valueExtractor(keyB)[quartilesKeys[4]]\n   * )},\n   */\n  boxwhisker.sortingFunction = function(_) { return arguments.length ? (sortingFunction = _, boxwhisker) : sortingFunction; };\n  /**\n   * Gets / sets the scale for which the boxwhisker values should be transformed by\n   * (see {@link boxwhisker#scale})\n   * @param {d3.scale} [_=none]\n   * @returns {boxwhisker | d3.scale}\n   * @memberof boxwhisker\n   * @property\n   * by default scale = d3.scaleLinear()\n   */\n  boxwhisker.scale = function(_) { return arguments.length ? (scale = _, boxwhisker) : scale; };\n  /**\n   * Gets / sets the padding for the domain of the scale\n   * (see {@link boxwhisker#domainPadding})\n   * @param {number} [_=none]\n   * @returns {boxwhisker | number}\n   * @memberof boxwhisker\n   * @property\n   * by default domainPadding = 0.5\n   */\n  boxwhisker.domainPadding = function(_) { return arguments.length ? (domainPadding = _, boxwhisker) : domainPadding; };\n  /**\n   * Gets / sets objectSpacer\n   * (see {@link boxwhisker#objectSpacer})\n   * @param {number} [_=none]\n   * @returns {boxwhisker | number}\n   * @memberof boxwhisker\n   * @property\n   * by default objectSpacer = 0.05\n   */\n  boxwhisker.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, boxwhisker) : objectSpacer; };\n  /**\n   * Gets / sets the minObjectSize\n   * (see {@link boxwhisker#minObjectSize})\n   * @param {number} [_=none]\n   * @returns {boxwhisker | number}\n   * @memberof boxwhisker\n   * @property\n   * by default minObjectSize = 15\n   */\n  boxwhisker.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, boxwhisker) : minObjectSize; };\n  /**\n   * Gets / sets the maxObjectSize\n   * (see {@link boxwhisker#maxObjectSize})\n   * @param {number} [_=none]\n   * @returns {boxwhisker | number}\n   * @memberof boxwhisker\n   * @property\n   * by default maxObjectSize = 50\n   */\n  boxwhisker.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, boxwhisker) : maxObjectSize; };\n  /**\n   * Gets / sets the whiskerWidthPercent\n   * (see {@link boxwhisker#whiskerWidthPercent})\n   * @param {number} [_=none]\n   * @returns {boxwhisker | number}\n   * @memberof boxwhisker\n   * @property\n   * by default maxObjectSize = 0.6\n   */\n  boxwhisker.whiskerWidthPercent = function(_) { return arguments.length ? (whiskerWidthPercent = _, boxwhisker) : whiskerWidthPercent; };\n  /**\n   * Gets / sets the colorFunction\n   * (see {@link boxwhisker#colorFunction})\n   * @param {colorFunction} [_=none]\n   * @returns {boxwhisker | colorFunction}\n   * @memberof boxwhisker\n   * @property\n   * by default colorFunction = colorFunction()\n   */\n  boxwhisker.colorFunction = function(_) { return arguments.length ? (colorFunction = _, boxwhisker) : colorFunction; };\n  /**\n   * Gets / sets the boxStrokeWidth\n   * (see {@link boxwhisker#boxStrokeWidth})\n   * @param {number} [_=none]\n   * @returns {boxwhisker | number}\n   * @memberof boxwhisker\n   * @property\n   * by default boxStrokeWidth = 2\n   */\n  boxwhisker.boxStrokeWidth = function(_) { return arguments.length ? (boxStrokeWidth = _, boxwhisker) : boxStrokeWidth; };\n  /**\n   * Gets / sets the whiskerStrokeWidth\n   * (see {@link boxwhisker#whiskerStrokeWidth})\n   * @param {number} [_=none]\n   * @returns {boxwhisker | number}\n   * @memberof boxwhisker\n   * @property\n   * by default whiskerStrokeWidth = 2\n   */\n  boxwhisker.whiskerStrokeWidth = function(_) { return arguments.length ? (whiskerStrokeWidth = _, boxwhisker) : whiskerStrokeWidth; };\n\n  /**\n   * Gets / sets the backgroundFill\n   * (see {@link boxwhisker#backgroundFill})\n   * @param {string} [_=none]\n   * @returns {boxwhisker | string}\n   * @memberof boxwhisker\n   * @property\n   * by default backgroundFill = 'transparent'\n   */\n  boxwhisker.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, boxwhisker) : backgroundFill; };\n  /**\n   * Gets / sets the namespace\n   * (see {@link boxwhisker#namespace})\n   * @param {string} [_=none]\n   * @returns {boxwhisker | string}\n   * @memberof boxwhisker\n   * @property\n   * by default namespace = 'd3sm-boxwhisker'\n   */\n  boxwhisker.namespace = function(_) { return arguments.length ? (namespace = _, boxwhisker) : namespace; };\n  /**\n   * Gets / sets the objectClass\n   * (see {@link boxwhisker#objectClass})\n   * @param {string} [_=none]\n   * @returns {boxwhisker | string}\n   * @memberof boxwhisker\n   * @property\n   * by default objectClass = 'tick-group'\n   */\n  boxwhisker.objectClass = function(_) { return arguments.length ? (objectClass = _, boxwhisker) : objectClass; };\n  /**\n   * Gets / sets the transitionDuration\n   * (see {@link boxwhisker#transitionDuration})\n   * @param {number} [_=none]\n   * @returns {boxwhisker | number}\n   * @memberof boxwhisker\n   * @property\n   * by default transitionDuration = 1000\n   */\n  boxwhisker.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, boxwhisker) : transitionDuration; };\n  /**\n   * Gets / sets the easeFunc\n   * (see {@link boxwhisker#easeFunc})\n   * @param {d3.ease} [_=none]\n   * @returns {boxwhisker | d3.ease}\n   * @memberof boxwhisker\n   * @property\n   * by default easeFunc = d3.easeExp\n   */\n  boxwhisker.easeFunc = function(_) { return arguments.length ? (easeFunc = _, boxwhisker) : easeFunc; };\n\n  /**\n   * Gets / sets the barKeys\n   * (see {@link boxwhisker#boxKeys})\n   * @param {string[]} [_=none]\n   * @returns {boxwhisker | string[]}\n   * @memberof boxwhisker\n   * @property\n   * by default boxKeys = undefined\n   */\n  boxwhisker.boxKeys = function(_) { return arguments.length ? (boxKeys = _, boxwhisker) : boxKeys; };\n  /**\n   * Gets / sets the boxValues\n   * (see {@link boxwhisker#boxValues})\n   * @param {number[]} [_=none]\n   * @returns {boxwhisker | number[]}\n   * @memberof boxwhisker\n   * @property\n   * by default boxValues = undefined\n   */\n  boxwhisker.boxValues = function(_) { return arguments.length ? (boxValues = _, boxwhisker) : boxValues; };\n  /**\n   * Gets / sets the objectSize\n   * (see {@link boxwhisker#objectSize})\n   * @param {number} [_=none]\n   * @returns {boxwhisker | number}\n   * @memberof boxwhisker\n   * @property\n   * by default objectSize = undefined\n   */\n  boxwhisker.objectSize = function(_) { return arguments.length ? (objectSize = _, boxwhisker) : objectSize; };\n  /**\n   * Gets / sets the spacerSize\n   * (see {@link boxwhisker#spacerSize})\n   * @param {number} [_=none]\n   * @returns {boxwhisker | number}\n   * @memberof boxwhisker\n   * @property\n   * by default spacerSize = undefined\n   */\n  boxwhisker.spacerSize = function(_) { return arguments.length ? (spacerSize = _, boxwhisker) : spacerSize; };\n  /**\n   * Gets / sets the tooltip\n   * (see {@link boxwhisker#tooltip})\n   * @param {tooltip} [_=none]\n   * @returns {boxwhisker | tooltip}\n   * @memberof boxwhisker\n   * @property\n   * by default tooltip = tooltip()\n   */\n  boxwhisker.tooltip = function(_) { return arguments.length ? (tooltip = _, boxwhisker) : tooltip; };\n\n\n  function boxwhisker() {\n    // for convenience in handling orientation specific values\n    var horizontalQ = (orient == 'horizontal') ? true : false\n    var verticalQ = !horizontalQ\n\n    // background cliping rectangle\n    var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}\n    var container = setupContainer( selection, namespace, bgcpRect, backgroundFill );\n\n    // if grouping is undefined sort keys by sorting funct\n    var ordered = (grouping == undefined) ? d3.keys(data).sort(sortingFunction) : grouping\n    // to prevent re-calculation and getters to be passed to axes\n    boxKeys = flatten(ordered)\n    boxValues = boxKeys.map(valueExtractor)\n\n\n    var numberOfObjects = boxKeys.length\n    var extent = [\n      Math.min(...boxValues.map(function(d,i){return d[quartilesKeys[0]]})) - domainPadding,\n      Math.max(...boxValues.map(function(d,i){return d[quartilesKeys[4]]})) + domainPadding\n    ];\n\n    // set the scale\n    scale.domain(extent).range(horizontalQ ? [0,spaceY] : [spaceX, 0])\n    var space = horizontalQ ? spaceX : spaceY\n    // calculate object size\n    objectSize = calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ)\n    // calculate spacer size if needed\n    spacerSize = calculateWidthOfSpacer(boxKeys, space, objectSize, numberOfObjects, objectSpacer, overflowQ)\n    // make the nested groups\n    var spacerFunction = groupingSpacer()\n    .horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects)\n    .objectClass(objectClass).objectSize(objectSize).spacerSize(spacerSize)\n    .transitionDuration(transitionDuration).easeFunc(easeFunc)\n    .namespace(namespace)\n\n    // move stuff\n    spacerFunction(container, ordered, 0)\n\n    var parentIndexArray = []\n    container.selectAll('g:not(.to-remove).'+objectClass)\n    .each(function(d, i){if (hasQ(boxKeys, d)){ parentIndexArray.push(Number(d3.select(this).attr('parent-index')))}})\n\n\n\n    colorFunction = colorFunction.colorBy() == 'index'\n    ? colorFunction.dataExtent([0, Math.max(...parentIndexArray)])\n    : colorFunction.dataExtent(extent)\n\n\n    // set attributes for box and whiskers\n    container.selectAll('g:not(.to-remove).'+objectClass).each(function(key, i) {\n      var t = d3.select(this),\n      currentData = data[key],\n\n      quartiles = valueExtractor(key, i),\n      q0 = quartiles[quartilesKeys[0]],\n      q1 = quartiles[quartilesKeys[1]],\n      q2 = quartiles[quartilesKeys[2]],\n      q3 = quartiles[quartilesKeys[3]],\n      q4 = quartiles[quartilesKeys[4]]\n\n      var i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'),\n      fillColor = colorFunction(key, q2, i, 'fill'), // prevent duplicate computation\n      strokeColor = colorFunction(key, q2, i,  'stroke')\n\n\n      var\n      whisk = safeSelect(t, 'g', 'whisker'),\n      uWhisk = safeSelect(whisk, 'path', 'upper'),\n      lWhisk = safeSelect(whisk, 'path', 'lower'),\n      quart = safeSelect(t, 'g', 'quartile'),\n      uQuart = safeSelect(quart, 'rect', 'upper'),\n      lQuart = safeSelect(quart, 'rect', 'lower'),\n      mQuart = safeSelect(quart, 'circle', 'median')\n\n\n      // set upper quartile (q3)\n      uQuart.transition().duration(transitionDuration).ease(easeFunc)\n      .attr('width', horizontalQ ? objectSize : scale(q3) - scale(q2))\n      .attr('height', verticalQ ? objectSize : scale(q3) - scale(q2))\n      .attr('fill', fillColor)\n      .attr('stroke', strokeColor)\n      .attr('stroke-width', boxStrokeWidth)\n      .attr('transform', function(d, i){\n        var\n        x = horizontalQ ? 0 : scale(q2),\n        y = verticalQ ? 0 : scale(extent[1]) - scale(q3),\n        t = 'translate('+x+','+y+')'\n        return t\n      })\n\n      // set lower quartile (q1)\n      lQuart.transition().duration(transitionDuration).ease(easeFunc)\n      .attr('width', horizontalQ ? objectSize : scale(q2) - scale(q1))\n      .attr('height', verticalQ ? objectSize : scale(q2) - scale(q1))\n      .attr('fill', fillColor)\n      .attr('stroke', strokeColor)\n      .attr('stroke-width', boxStrokeWidth)\n      .attr('transform', function(d, i){\n        var\n        x = horizontalQ ? 0 : scale(q1),\n        y = verticalQ ? 0 : scale(extent[1]) - scale(q2),\n        t = 'translate('+x+','+y+')'\n        return t\n      })\n\n\n      // set median (q2)\n      mQuart.transition().duration(transitionDuration).ease(easeFunc)\n      .attr('r', function(d, i){\n        var r = objectSize / 2\n        var dif = (scale(q3) - scale(q1)) / 2\n        return (r > dif) ? dif : r\n      })\n      .attr('fill', fillColor)\n      .attr('stroke', strokeColor)\n      .attr('stroke-width', boxStrokeWidth)\n      .attr('transform', function(d, i){\n        var\n        x = horizontalQ ? objectSize / 2 : scale(q2),\n        y = verticalQ ? objectSize / 2 : scale(extent[1]) - scale(q2),\n        t = 'translate('+x+','+y+')'\n        return t\n      })\n\n      // set lower whisker (min)\n      lWhisk.transition().duration(transitionDuration).ease(easeFunc)\n      .attr('d', function(dd, ii){\n        var\n        dir = false,\n        x = 0,\n        y = 0,\n        h = horizontalQ ? scale(q1) - scale(q0) : objectSize,\n        w = verticalQ ? scale(q1) - scale(q0) : objectSize\n        return whiskerPath(dir, x, y, w, h, whiskerWidthPercent, orient)\n      })\n      .attr('transform', function(d, i){\n        var\n        x = horizontalQ ? 0 : scale(q1),\n        y = verticalQ ? 0 : scale(extent[1]) - scale(q1),\n        t = 'translate('+x+','+y+')'\n        return t\n      })\n      .attr('stroke', 'black').attr('stroke-width', whiskerStrokeWidth)\n      .attr('fill', 'none')\n\n      // set upper whisker (max)\n      uWhisk.transition().duration(transitionDuration).ease(easeFunc)\n      .attr('d', function(dd, ii){\n        var\n        dir = true,\n        x = 0,\n        y = 0,\n        h = horizontalQ ? scale(q4) - scale(q3) : objectSize,\n        w = verticalQ ? scale(q4) - scale(q3) : objectSize\n        return whiskerPath(dir, x, y, w, h, whiskerWidthPercent, orient)\n      })\n      .attr('transform', function(d, i){\n        var\n        x = horizontalQ ? 0 : scale(q3),\n        y = verticalQ ? 0 :  scale(extent[1]) - scale(q4),\n        t = 'translate('+x+','+y+')'\n        return t\n      })\n      .attr('stroke', 'black')\n      .attr('stroke-width', whiskerStrokeWidth)\n      .attr('fill', 'none')\n\n    })\n\n    tooltip.selection(container.selectAll('g:not(.to-remove).'+objectClass))\n    .data(data)\n    tooltip()\n\n\n  }\n\n  return boxwhisker\n}\n","import {hypenate, safeSelect} from './helpers';\nimport {selectFilter} from './select-filter';\n\n/*******************************************************************************\n**                                                                            **\n**                                                                            **\n**                                DATATOGGLE                                  **\n**                                                                            **\n**                                                                            **\n*******************************************************************************/\n/**\n * Creates a datatoggle\n * @constructor datatoggle\n * @param {d3.selection} selection\n * @namespace datatoggle\n * @returns {function} datatoggle\n */\nexport function datatoggle( selection ) {\n  var\n  /**\n  * Keys to make toggle-able options\n  * (see {@link datatoggle#keys})\n  * @param {string[]} [keys=undefined]\n  * @memberof datatoggle#\n  * @property\n  */\n  keys,\n  /**\n  * What to do when a different key is clicked\n  * (see {@link datatoggle#updateFunction})\n  * @param {function} [updateFunction=function(){}]\n  * @memberof datatoggle#\n  * @property\n  */\n  updateFunction = function(){},\n  /**\n  * Namespace for all items made by this instance of datatoggle\n  * @param {string} [namespace=\"d3sm-databar\"]\n  * @memberof datatoggle#\n  * @property\n  */\n  namespace='d3sm-databar',\n  /**\n  * Currently toggled key\n  * @param {string} [currentKey=undefined]\n  * @memberof datatoggle#\n  * @property\n  */\n  currentKey,\n\n  xAxisSelectQ = false,\n  xAxisOptions,\n  yAxisSelectQ = false,\n  yAxisOptions,\n  data\n  toggle.xAxisSelectQ = function(_){return arguments.length ? (xAxisSelectQ = _, toggle) : xAxisSelectQ}\n  toggle.yAxisSelectQ = function(_){return arguments.length ? (yAxisSelectQ = _, toggle) : yAxisSelectQ}\n  toggle.xAxisOptions = function(_){return arguments.length ? (xAxisOptions = _, toggle) : xAxisOptions}\n  toggle.yAxisOptions = function(_){return arguments.length ? (yAxisOptions = _, toggle) : yAxisOptions}\n  toggle.data = function(_){return arguments.length ? (data = _, toggle) : data}\n\n\n  /**\n   * Gets / sets the updateFunction\n   * (see {@link datatoggle#updateFunction})\n   * @function datatoggle.updateFunction\n   * @param {function} [_=none]\n   * @returns {datatoggle | function}\n   * @memberof datatoggle\n   * @property\n   * by default updateFunction = function(){}\n   */\n  toggle.updateFunction = function(_){return arguments.length ? (updateFunction = _, toggle) : updateFunction}\n  /**\n   * Gets / sets the namespace\n   * (see {@link datatoggle#namespace})\n   * @function datatoggle.namespace\n   * @param {string} [_=none]\n   * @returns {datatoggle | string}\n   * @memberof datatoggle\n   * @property\n   * by default namespace = 'd3sm-databar'\n   */\n  toggle.namespace = function(_){return arguments.length ? (namespace = _, toggle) : namespace}\n  /**\n   * Gets / sets the currentKey\n   * (see {@link datatoggle#currentKey})\n   * @function datatoggle.currentKey\n   * @param {string} [_=none]\n   * @returns {datatoggle | string}\n   * @memberof datatoggle\n   * @property\n   * by default currentKey = undefined\n   */\n  toggle.currentKeys =\n  function () {\n    var vals = {}\n    d3.keys(filterSelects).map(function(k, i){\n      vals[k]= filterSelects[k].currentOption()\n    })\n    return vals\n  }\n\n  var filterSelects = {}\n  function toggle() {\n    // selection options\n\n    // selection.classed('d-flex flex-row', true)\n    // var filterButton = safeSelect(selection, 'a', 'slider-buttons')\n    // var filterI = safeSelect(filterButton, 'i', 'fa fa-sliders')\n\n    /*BUG: unexpected behavior.\n      - when using bootstrap-eque way for collapse, clicking button submits to\n      the same page in applications (but not in demo), so using anchor (<a>)\n      - when using anchor, cause page to jump to that location\n      - when using show, the first open works, but the close does not.\n    */\n    // filterButton.attr('class', 'btn btn-secondary')\n    // .attr('data-toggle', 'collapse')\n    // .attr('href', '#'+hypenate(namespace, 'data-toggle'))\n    // .attr('target', '_blank')\n    // .html(filterButton.html()+' Filters')\n    // .on('click', function(d, i){\n    //   d3.event.preventDefault()\n    //   d3.event.stopPropagation()\n    //   var dt = d3.select(\"#\"+hypenate(namespace, 'data-toggle'))\n    //   dt.classed('show', !dt.classed('show'))\n    //   dt.classed('d-inline-flex', dt.classed('show'))\n    //\n    //   filterButton.classed('btn-primary', dt.classed('show'))\n    //   filterButton.classed('btn-secondary', !dt.classed('show'))\n    // })\n    // .classed('d-inline-flex', true)\n    // .style(\"margin-right\", '10px')\n\n\n\n    // var datatoggleCollapse = safeSelect(selection, 'div', hypenate(namespace,'collapse'))\n    // .attr('id', hypenate(namespace, 'data-toggle'))\n    // .classed('collapse', true)\n\n    // var flexRow = safeSelect(datatoggleCollapse, 'div', 'd-inline-flex flex-row flex-wrap')\n    var flexRow = safeSelect(selection, 'div', 'd-inline-flex flex-row flex-wrap')\n\n    var dataopts = flexRow.selectAll('div.'+hypenate(namespace,'select-filter'))\n    // remove excess\n    dataopts.exit().remove()\n    // bind data\n    dataopts = dataopts.data(d3.keys(data))\n    //enter\n    var doEnter = dataopts.enter().append('div')\n    .attr('class', 'select-filter')\n\n    dataopts = dataopts.merge(doEnter).style('margin-right', \"10px\")\n\n    dataopts.each(function(d, i){\n      var t = d3.select(this)\n      var sf = selectFilter(t)\n      .data(data[d])\n      .namespace(hypenate(namespace, d))\n      .selectionName(d)\n      sf()\n      filterSelects[d] = sf\n    })\n\n\n    selection.selectAll('select')\n    .on('change', function(){updateFunction()})\n    // bind update function\n    // d3.selectAll\n    return toggle\n  }\n\n\n  function onlyOne() {\n    // d3.event.preventDefault()\n    // d3.event.stopPropagation()\n    var dataopts = selection.selectAll('div.data-option')\n    currentKey = dataopts.select(':checked').datum()\n    updateFunction()\n\n  }\n\n  function axisSelectFilter(selection, axis=\"x-axis\", axisData) {\n    if (axisData == undefined) {\n      axisData = {\n        'linear': d3.scaleLinear(),\n        'log': d3.scaleLog(),\n        'pow': d3.scalePow(),\n        'sqrt': d3.scaleSqrt()\n      }\n    }\n\n    var sf = selectFilter(selection)\n    .data(axisData)\n    .namespace(axis)\n    .selectionName(axis+' scale')\n    .defaultValue(d3.scaleLinear())\n\n\n    sf()\n\n  }\n\n  return toggle\n}\n","import {hypenate, safeSelect} from './helpers';\nimport {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer} from './utils';\nimport {unique, flatten} from './array-functions';\nimport {groupingSpacer} from './grouping-spacer';\nimport {colorFunction as CF} from './color-function';\nimport {tooltip as TTip} from './tooltip';\n/*******************************************************************************\n**                                                                            **\n**                                                                            **\n**                                 SCATTER                                    **\n**                                                                            **\n**                                                                            **\n*******************************************************************************/\n/**\n * Creates a scatter\n *\n * {@link https://sumneuron.gitlab.io/d3sm/demos/scatter/index.html Demo}\n * @constructor scatter\n * @param {d3.selection} selection\n * @namespace scatter\n * @returns {function} scatter\n */\nexport function scatter ( selection ) {\n\n  var\n  /**\n  * Data to plot. Assumed to be a object, where each key corresponds to a point\n  * (see {@link scatter#data})\n  * @param {Object} [data=undefined]\n  * @memberof scatter#\n  * @property\n  */\n  data,\n  /**\n  * Amount of horizontal space (in pixels) avaible to render the scatter in\n  * (see {@link scatter#spaceX})\n  * @param {number} [spaceX=undefined]\n  * @memberof scatter#\n  * @property\n  */\n  spaceX,\n  /**\n  * Amount of vertical space (in pixels) avaible to render the scatter in\n  * (see {@link scatter.spaceY})\n  * @param {number} [spaceY=undefined]\n  * @memberof scatter#\n  * @property\n  */\n  spaceY,\n\n  /**\n  * The scale for which scatter x values should be transformed by\n  * @param {d3.scale} [scaleX=d3.scaleLinear]\n  * @memberof scatter#\n  * @property\n  */\n  scaleX = d3.scaleLinear(),\n  /**\n  * The padding for the domain of the scaleX (see {@link scatter#scaleX})\n  * @param {number} [domainPaddingX=0.5]\n  * @memberof scatter#\n  * @property\n  */\n  domainPaddingX = 0.5,\n  /**\n  * The function for getting the x value of the current point\n  * @param {function} [valueExtractorX=function(d, i){return data[d]['x']}]\n  * @memberof scatter#\n  * @property\n  */\n  valueExtractorX = function(d, i) {return data[d]['x']},\n\n  /**\n  * The scale for which scatter y values should be transformed by\n  * @param {d3.scale} [scaleY=d3.scaleLinear]\n  * @memberof scatter#\n  * @property\n  */\n  scaleY = d3.scaleLinear(),\n  /**\n  * The padding for the domain of the scaleY (see {@link scatter#scaleY})\n  * @param {number} [domainPaddingY=0.5]\n  * @memberof scatter#\n  * @property\n  */\n  domainPaddingY = 0.5,\n  /**\n  * The function for getting the y value of the current point\n  * @param {function} [valueExtractorY=function(d, i){return data[d]['y']}]\n  * @memberof scatter#\n  * @property\n  */\n  valueExtractorY = function(d, i) {return data[d]['y']},\n\n\n  /**\n  * The scale for which scatter r values should be transformed by\n  * @param {d3.scale} [scaleR=d3.scaleLinear]\n  * @memberof scatter#\n  * @property\n  */\n  scaleR = d3.scaleLinear(),\n  /**\n  * The padding for the domain of the scaleR (see {@link scatter#scaleR})\n  * @param {number} [domainPaddingR=0.5]\n  * @memberof scatter#\n  * @property\n  */\n  domainPaddingR = 0.5,\n  /**\n  * The function for getting the r value of the current point\n  * @param {function} [valueExtractorR=function(d, i){return 2}]\n  * @memberof scatter#\n  * @property\n  */\n  valueExtractorR = function(d, i) {return 2},\n  /**\n  * The min radius a point can have\n  * @param {function} [minRadius=2]\n  * @memberof scatter#\n  * @property\n  */\n  minRadius = 2,\n  /**\n  * The min radius a point can have\n  * @param {function} [maxRadius=10]\n  * @memberof scatter#\n  * @property\n  */\n  maxRadius = 10,\n\n  /**\n  * The stroke width of the points\n  * @param {number} [pointStrokeWidth=2]\n  * @memberof scatter#\n  * @property\n  */\n  pointStrokeWidth = 2,\n  /**\n  * Instance of ColorFunction\n  * @param {function} [colorFunction = colorFunction()]\n  * @memberof scatter#\n  * @property\n  */\n  colorFunction = CF(),\n  /**\n  * Color of the background\n  * @param {string} [backgroundFill=\"transparent\"]\n  * @memberof scatter#\n  * @property\n  */\n  backgroundFill = 'transparent',\n  /**\n  * Namespace for all items made by this instance of scatter\n  * @param {string} [namespace=\"d3sm-scatter\"]\n  * @memberof scatter#\n  * @property\n  */\n  namespace = 'd3sm-scatter',\n  /**\n  * Class name for scatter container (<circle> element)\n  * @param {string} [objectClass=\"scatter-point\"]\n  * @memberof scatter#\n  * @property\n  */\n  objectClass = 'scatter-point',\n  /**\n  * Duration of all transitions of this element\n  * @param {number} [transitionDuration=1000]\n  * @memberof scatter#\n  * @property\n  */\n  transitionDuration = 1000,\n  /**\n  * Easing function for transitions\n  * @param {d3.ease} [easeFunc=d3.easeExp]\n  * @memberof scatter#\n  * @property\n  */\n  easeFunc = d3.easeExp,\n\n  // useful values to extract to prevent re-calculation\n  /**\n  * The keys of the points\n  * @param {string[]} [pointKeys=undefined]\n  * @memberof scatter#\n  * @property\n  */\n  pointKeys,\n  /**\n  * The x values of the points\n  * @param {number[]} [valuesX=undefined]\n  * @memberof scatter#\n  * @property\n  */\n  valuesX,\n  /**\n  * The y values of the points\n  * @param {number[]} [valuesY=undefined]\n  * @memberof scatter#\n  * @property\n  */\n  valuesY,\n  /**\n  * The r values of the points\n  * @param {number[]} [valuesR=undefined]\n  * @memberof scatter#\n  * @property\n  */\n  valuesR,\n\n  /**\n  * Instance of Tooltip\n  * @param {function} [tooltip=tooltip()]\n  * @memberof scatter#\n  * @property\n  */\n  tooltip = TTip()\n\n  /**\n   * Gets or sets the selection in which items are manipulated\n   * @param {d3.selection} [_=none]\n   * @returns {scatter | d3.selection}\n   * @memberof scatter\n   * @property\n   * by default selection = selection\n   */\n  scatter.selection = function(_) { return arguments.length ? (selection =_, scatter) : selection}\n  /**\n   * Gets or sets the data\n   * (see {@link scatter#data})\n   * @param {number} [_=none]\n   * @returns {scatter | object}\n   * @memberof scatter\n   * @property\n   */\n  scatter.data = function(_) { return arguments.length ? (data =_, scatter) : data}\n  /**\n   * Gets or sets the amount of horizontal space in which items are manipulated\n   * (see {@link scatter#spaceX})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {scatter | number}\n   * @memberof scatter\n   * @property\n   * by default spaceX = undefined\n   */\n  scatter.spaceX = function(_) { return arguments.length ? (spaceX =_, scatter) : spaceX}\n  /**\n   * Gets or sets the amount of vertical space in which items are manipulated\n   * (see {@link scatter#spaceY})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {scatter | number}\n   * @memberof scatter\n   * @property\n   * by default spaceY = undefined\n   */\n  scatter.spaceY = function(_) { return arguments.length ? (spaceY =_, scatter) : spaceY}\n\n\n\n  /**\n   * Gets / sets the x scale for which the scatter x values should be transformed by\n   * (see {@link scatter#scaleX})\n   * @param {d3.scale} [_=none]\n   * @returns {scatter | d3.scale}\n   * @memberof scatter\n   * @property\n   * by default scaleX = d3.scaleLinear()\n   */\n  scatter.scaleX = function(_) { return arguments.length ? (scaleX =_, scatter) : scaleX}\n  /**\n   * Gets / sets the padding for the domain of the x scale\n   * (see {@link scatter#domainPaddingX})\n   * @param {number} [_=none]\n   * @returns {scatter | number}\n   * @memberof scatter\n   * @property\n   * by default domainPaddingX = 0.5\n   */\n  scatter.domainPaddingX = function(_) { return arguments.length ? (domainPaddingX =_, scatter) : domainPaddingX}\n  /**\n   * Gets / sets the valueExtractorX\n   * (see {@link scatter#valueExtractorX})\n   * @param {function} [_=none]\n   * @returns {scatter | function}\n   * @memberof scatter\n   * @property\n   * by default valueExtractorX = function(key, index) { return data[key]['x'] }\n   */\n  scatter.valueExtractorX = function(_) { return arguments.length ? (valueExtractorX =_, scatter) : valueExtractorX}\n\n\n  /**\n   * Gets / sets the y scale for which the scatter y values should be transformed by\n   * (see {@link scatter#scaleY})\n   * @param {d3.scale} [_=none]\n   * @returns {scatter | d3.scale}\n   * @memberof scatter\n   * @property\n   * by default scaleY = d3.scaleLinear()\n   */\n  scatter.scaleY = function(_) { return arguments.length ? (scaleY =_, scatter) : scaleY}\n  /**\n   * Gets / sets the padding for the domain of the y scale\n   * (see {@link scatter#domainPaddingY})\n   * @param {number} [_=none]\n   * @returns {scatter | number}\n   * @memberof scatter\n   * @property\n   * by default domainPaddingY = 0.5\n   */\n  scatter.domainPaddingY = function(_) { return arguments.length ? (domainPaddingY =_, scatter) : domainPaddingY}\n  /**\n   * Gets / sets the valueExtractorY\n   * (see {@link scatter#valueExtractorY})\n   * @param {function} [_=none]\n   * @returns {scatter | function}\n   * @memberof scatter\n   * @property\n   * by default valueExtractorY = function(key, index) { return data[key]['y'] }\n   */\n  scatter.valueExtractorY = function(_) { return arguments.length ? (valueExtractorY =_, scatter) : valueExtractorY}\n\n\n  /**\n   * Gets / sets the r scale for which the scatter r values should be transformed by\n   * (see {@link scatter#scaleR})\n   * @param {d3.scale} [_=none]\n   * @returns {scatter | d3.scale}\n   * @memberof scatter\n   * @property\n   * by default scaleR = d3.scaleLinear()\n   */\n  scatter.scaleR = function(_) { return arguments.length ? (scaleR =_, scatter) : scaleR}\n  /**\n   * Gets / sets the padding for the domain of the r scale\n   * (see {@link scatter#domainPaddingR})\n   * @param {number} [_=none]\n   * @returns {scatter | number}\n   * @memberof scatter\n   * @property\n   * by default domainPaddingR = 0.5\n   */\n  scatter.domainPaddingR = function(_) { return arguments.length ? (domainPaddingR =_, scatter) : domainPaddingR}\n  /**\n   * Gets / sets the valueExtractorR\n   * (see {@link scatter#valueExtractorR})\n   * @param {function} [_=none]\n   * @returns {scatter | function}\n   * @memberof scatter\n   * @property\n   * by default valueExtractorR = function(key, index) { return data[key]['r'] }\n   */\n  scatter.valueExtractorR = function(_) { return arguments.length ? (valueExtractorR =_, scatter) : valueExtractorR}\n  /**\n   * Gets / sets the minRadius\n   * (see {@link scatter#minRadius})\n   * @param {number} [_=none]\n   * @returns {scatter | number}\n   * @memberof scatter\n   * @property\n   * by default minRadius = 2\n   */\n  scatter.minRadius = function(_) { return arguments.length ? (minRadius =_, scatter) : minRadius}\n  /**\n   * Gets / sets the maxRadius\n   * (see {@link scatter#maxRadius})\n   * @param {number} [_=none]\n   * @returns {scatter | number}\n   * @memberof scatter\n   * @property\n   * by default maxRadius = 10\n   */\n  scatter.maxRadius = function(_) { return arguments.length ? (maxRadius =_, scatter) : maxRadius}\n\n  /**\n   * Gets / sets the pointStrokeWidth\n   * (see {@link scatter#pointStrokeWidth})\n   * @param {number} [_=none]\n   * @returns {scatter | number}\n   * @memberof scatter\n   * @property\n   * by default pointStrokeWidth = 2\n   */\n  scatter.pointStrokeWidth = function(_) { return arguments.length ? (pointStrokeWidth =_, scatter) : pointStrokeWidth}\n  /**\n   * Gets / sets the colorFunction\n   * (see {@link scatter#colorFunction})\n   * @param {number} [_=none]\n   * @returns {scatter | number}\n   * @memberof scatter\n   * @property\n   * by default colorFunction = colorFunction()\n   */\n  scatter.colorFunction = function(_) { return arguments.length ? (colorFunction =_, scatter) : colorFunction}\n  /**\n   * Gets / sets the backgroundFill\n   * (see {@link scatter#backgroundFill})\n   * @param {string} [_=none]\n   * @returns {scatter | string}\n   * @memberof scatter\n   * @property\n   * by default backgroundFill = 'transparent'\n   */\n  scatter.backgroundFill = function(_) { return arguments.length ? (backgroundFill =_, scatter) : backgroundFill}\n  /**\n   * Gets / sets the namespace\n   * (see {@link scatter#namespace})\n   * @param {string} [_=none]\n   * @returns {scatter | string}\n   * @memberof scatter\n   * @property\n   * by default namespace = 'd3sm-scatter'\n   */\n  scatter.namespace = function(_) { return arguments.length ? (namespace =_, scatter) : namespace}\n  /**\n   * Gets / sets the objectClass\n   * (see {@link scatter#objectClass})\n   * @param {string} [_=none]\n   * @returns {scatter | string}\n   * @memberof scatter\n   * @property\n   * by default objectClass = 'tick-group'\n   */\n  scatter.objectClass = function(_) { return arguments.length ? (objectClass =_, scatter) : objectClass}\n  /**\n   * Gets / sets the transitionDuration\n   * (see {@link scatter#transitionDuration})\n   * @param {number} [_=none]\n   * @returns {scatter | number}\n   * @memberof scatter\n   * @property\n   * by default transitionDuration = 1000\n   */\n  scatter.transitionDuration = function(_) { return arguments.length ? (transitionDuration =_, scatter) : transitionDuration}\n  /**\n   * Gets / sets the easeFunc\n   * (see {@link scatter#easeFunc})\n   * @param {d3.ease} [_=none]\n   * @returns {scatter | d3.ease}\n   * @memberof scatter\n   * @property\n   * by default easeFunc = d3.easeExp\n   */\n  scatter.easeFunc = function(_) { return arguments.length ? (easeFunc =_, scatter) : easeFunc}\n\n  /**\n   * Gets / sets the pointKeys\n   * (see {@link scatter#pointKeys})\n   * @param {string[]} [_=none]\n   * @returns {scatter | string[]}\n   * @memberof scatter\n   * @property\n   * by default pointKeys = undefined\n   */\n  scatter.pointKeys = function(_) { return arguments.length ? (pointKeys =_, scatter) : pointKeys}\n  /**\n   * Gets / sets the valuesX\n   * (see {@link scatter#valuesX})\n   * @param {number[]} [_=none]\n   * @returns {scatter | number[]}\n   * @memberof scatter\n   * @property\n   * by default valuesX = undefined\n   */\n  scatter.valuesX = function(_) { return arguments.length ? (valuesX =_, scatter) : valuesX}\n  /**\n   * Gets / sets the valuesY\n   * (see {@link scatter#valuesY})\n   * @param {number[]} [_=none]\n   * @returns {scatter | number[]}\n   * @memberof scatter\n   * @property\n   * by default valuesY = undefined\n   */\n  scatter.valuesY = function(_) { return arguments.length ? (valuesY =_, scatter) : valuesY}\n  /**\n   * Gets / sets the valuesR\n   * (see {@link scatter#valuesR})\n   * @param {number[]} [_=none]\n   * @returns {scatter | number[]}\n   * @memberof scatter\n   * @property\n   * by default valuesR = undefined\n   */\n  scatter.valuesR = function(_) { return arguments.length ? (valuesR =_, scatter) : valuesR}\n  /**\n   * Gets / sets the tooltip\n   * (see {@link scatter#tooltip})\n   * @param {tooltip} [_=none]\n   * @returns {scatter | tooltip}\n   * @memberof scatter\n   * @property\n   * by default tooltip = tooltip()\n   */\n\n  scatter.tooltip = function(_) { return arguments.length ? (tooltip =_, scatter) : tooltip}\n\n\n  function scatter() {\n    // background cliping rectangle\n    var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}\n    var container = setupContainer( selection, namespace, bgcpRect, backgroundFill );\n\n\n    pointKeys = d3.keys(data)\n    valuesX = pointKeys.map(valueExtractorX)\n    valuesY = pointKeys.map(valueExtractorY)\n    valuesR = pointKeys.map(valueExtractorR)\n\n    var numberOfObjects = pointKeys.length\n    var extentX = [Math.min(...valuesX) - domainPaddingX, Math.max(...valuesX) + domainPaddingX]\n    var extentY = [Math.min(...valuesY) - domainPaddingY, Math.max(...valuesY) + domainPaddingY]\n    var extentR = [Math.min(...valuesR) - domainPaddingR, Math.max(...valuesR) + domainPaddingR]\n\n    scaleX.domain(extentX).range([0, spaceX])\n    scaleY.domain(extentY).range([spaceY, 0])\n    scaleR.domain(extentR).range([minRadius, maxRadius])\n\n    var points = container.selectAll('.'+objectClass)\n    points = points.data(pointKeys)\n    var pEnter = points.enter().append('circle')\n    .attr('class', objectClass)\n    .attr('cx', 0).attr('cy', spaceY).attr('r', 0)\n\n    var pExit = points.exit()\n\n    points = points.merge(pEnter)\n\n    points.each(function(key, i){\n      var t = d3.select(this),\n      currentData = data[key],\n      x = valuesX[i],\n      y = valuesY[i],\n      r = valuesR[i],\n      fillColor = colorFunction(key, currentData, i, 'fill'),\n      strokeColor = colorFunction(key, currentData, i, 'stroke')\n\n      t.transition().duration(transitionDuration).ease(easeFunc)\n      .attr('cx', scaleX(x))\n      .attr('cy', scaleY(y))\n      .attr('r', scaleR(r))\n      .attr('fill', fillColor)\n      .attr('stroke', strokeColor)\n      .attr('stroke-width', pointStrokeWidth)\n\n\n\n      t.on('mouseover', function(d, i){\n        points.style('opacity', 0.2)\n        t.style('opacity', 1)\n        t.transition().duration(transitionDuration/2).ease(easeFunc)\n        .attr('stroke-width', pointStrokeWidth*2)\n        .attr('r', scaleR(r) * 1.5)\n\n      })\n      t.node().addEventListener('mouseout', function(){\n        container.selectAll('.'+objectClass).style('opacity', 1)\n        t.transition().duration(transitionDuration/2).ease(easeFunc)\n        .attr('stroke-width', pointStrokeWidth)\n        .attr('r', scaleR(r))\n\n      })\n\n    })\n\n\n\n    pExit.transition().duration(transitionDuration).ease(easeFunc)\n    .attr('cx', 0).attr('cy', spaceY).attr('r', 0)\n    .remove()\n\n    tooltip.selection(points)\n    .data(data)\n\n    tooltip()\n  }\n\n\n  return scatter\n}\n","import {hypenate, safeSelect, getTranslation} from './helpers';\nimport {log, warn, error, info} from './utils';\n/*******************************************************************************\n\n**                                                                            **\n**                                                                            **\n**                                PLOTZOOM                                    **\n**                                                                            **\n**                                                                            **\n*******************************************************************************/\n/**\n * Creates an plotZoom instance, which can handle drag and scroll events\n * @constructor plotZoom\n * @param {function} chart a function instance of one of the d3sm plots (e.g. bar, boxwhisker, bubbleHeatmap, violin, etc)\n * @param {axis}  xAxis the axis instance responsible for the x axis\n * @param {axis} yAxis the axis instance responsible for the y axis\n * @namespace plotZoom\n * @returns {function} zoom\n */\nexport function plotZoom( chart, xAxis, yAxis ) {\n  var\n  /**\n  * The event on which to fire\n  * (see {@link plotZoom#eventType})\n  * @param {string} eventType which event it should handle. Currently supports scroll and wheel\n  * @memberof plotZoom#\n  * @property\n  */\n  eventType,\n  /**\n  * A scaling factor for the wheel \"speed\"\n  * (see {@link plotZoom#wheelSpeed})\n  * @param {number} [wheelSpeed=20] scales the wheel translation by wheelSpeed\n  * @memberof plotZoom#\n  * @property\n  */\n  wheelSpeed = 20,\n  /**\n  * The orientation in which to allow scrolling: 'horizontal', 'vertical', or '2D'\n  * (see {@link plotZoom#orient})\n  * @param {string} [orient=chart.orient() || 'horizontal']\n  * @memberof plotZoom#\n  * @property\n  */\n  orient = (chart.orient == undefined) ? 'horizontal' : chart.orient(),\n  /**\n  * The max distance allowed to scroll in the x direction\n  * (see {@link plotZoom#xLock})\n  * @param {number} [xLock=chart.spaceX()] ideally chart.overflowQ() == true and this value is the\n  * bounding rect across all elements in the chart  minus the space in which to show.\n  * @memberof plotZoom#\n  * @property\n  */\n  xLock=chart.spaceX(),\n  /**\n  * The max distance allowed to scroll in the y direction\n  * (see {@link plotZoom#yLock})\n  * @param {number} [yLock=chart.spaceY()] ideally chart.overflowQ() == true and this value is the\n  * bounding rect across all elements in the chart  minus the space in which to show.\n  * @memberof plotZoom#\n  * @property\n  */\n  yLock=chart.spaceY(),\n\n  chartSel = chart.selection(),\n  xAxisSel = xAxis.selection(),\n  yAxisSel = yAxis.selection(),\n  svg = d3.select(chartSel.thisSVG())\n\n\n  /**\n   * Gets or sets the event type in which to respond\n   * (see {@link plotZoom#eventType})\n   * @param {string} [_=none] should be 'drag' or 'wheel'\n   * @returns {zoom | string}\n   * @memberof plotZoom\n   * @property\n   * by default plotZoom=undefined\n   */\n  zoom.eventType = function(_) { return arguments.length ? (eventType = _, zoom) : eventType; };\n  /**\n   * Gets or sets the wheel speed in which to scale the wheel scroll transform\n   * (see {@link plotZoom#wheelSpeed})\n   * @param {number} [_=none]\n   * @returns {zoom | number}\n   * @memberof plotZoom\n   * @property\n   * by default wheelSpeed=20\n   */\n  zoom.wheelSpeed = function(_) { return arguments.length ? (wheelSpeed = _, zoom) : wheelSpeed; };\n  /**\n   * Gets or sets the orientation in which items are manipulated\n   * (see {@link plotZoom#orient})\n   * @param {string} [_=none] should be horizontal, vertical, or 2D\n   * @returns {zoom | string}\n   * @memberof plotZoom\n   * @property\n   * by default orient=chart.orient() || 'horizontal'\n   */\n  zoom.orient = function(_) { return arguments.length ? (orient = _, zoom) : orient; };\n\n  /**\n   * Gets or sets the max distance in which to scroll X\n   * (see {@link plotZoom#xLock})\n   * @param {number} [_=none] should be a positive value\n   * @returns {zoom | number}\n   * @memberof plotZoom\n   * @property\n   * by default xLock=chart.spaceX()\n   */\n  zoom.xLock = function(_) { return arguments.length ? (xLock = _, zoom) : xLock; };\n  /**\n   * Gets or sets the max distance in which to scroll Y\n   * (see {@link plotZoom#yLock})\n   * @param {number} [_=none]  should be a positive value\n   * @returns {zoom | number}\n   * @memberof plotZoom\n   * @property\n   * by default yLock=chart.spaceY()\n   */\n  zoom.yLock = function(_) { return arguments.length ? (yLock = _, zoom) : yLock; };\n\n\n  function setLocks() {\n    var chartObjSel = chartSel.select('.'+hypenate(chart.namespace(),'object-container'))\n    var chartObjTrans = getTranslation(chartObjSel.attr('transform'))\n    var cos = chartObjSel.attr('transform', 'translate(0,0)')\n    xLock = chartSel.node().getBBox().width - chart.spaceX() * .9\n    yLock = chartSel.node().getBBox().height - chart.spaceY() * .9\n    cos.attr('transform', 'translate('+chartObjTrans[0]+','+chartObjTrans[1]+')')\n    log('plotZoom', 'setLocks', {xLock:xLock, yLock:yLock})\n  }\n\n\n  /**\n   * Sets the x and y locks (how far one can scroll)\n   * (see {@link plotZoom#xLock} and {@link plotZoom#yLock})\n   * @function plotZoom.setLocks\n   * @returns {undefined}\n   * @memberof plotZoom\n   * @property\n   */\n  zoom.setLocks = setLocks\n\n  function zoom() {\n    setLocks()\n\n    var horizontalQ, verticalQ\n    if (orient == '2D') {horizontalQ = true; verticalQ = true;}\n    if (orient == 'horizontal') {horizontalQ = true; verticalQ = false;}\n    if (orient == 'vertical') {verticalQ = true; horizontalQ = false;}\n\n    // capture transform event\n    var transform = d3.event.transform\n\n    var chartBox = chartSel.node().getBBox()\n    var xAxisBox = xAxisSel.node().getBBox()\n    var yAxisBox = xAxisSel.node().getBBox()\n\n    var chartWidth = chartBox.width - chartBox.x\n    var chartHeight = chartBox.height - chartBox.y\n    var xAxisWidth = xAxisBox.width// - xAxisBox.x\n    var xAxisHeight = xAxisBox.height// - xAxisBox.y\n    var yAxisWidth = yAxisBox.width// - yAxisBox.x\n    var yAxisHeight = yAxisBox.height// -yAxisBox.y\n\n    // enable wheel event\n    if (eventType == \"wheel\") {\n      var e = d3.event\n      // prevent page scrolling\n      e.preventDefault()\n      // event delta is very very slow, so speed it up with wheel speed\n      var w = d3.event.deltaY * wheelSpeed\n      var shiftQ = d3.event.shiftKey\n\n      // d3 has no way to make custom transform, so make an object and add\n      // the necessary functions to make wheel event compatible with drag events\n      if (orient == '2D') {\n        transform = shiftQ ? {k: 1, x: w, y: 0} : {k: 1, x: 0, y: w}\n      } else {\n        transform = horizontalQ ? {k: 1, x: w, y: 0} : {k: 1, x: 0, y: w}\n      }\n      // the * -1 inverts the direction\n      transform.applyX = function(x) { return x * this.k + this.x * -1; }\n      transform.applyY =  function(y) { return y * this.k + this.y * -1; }\n    }\n\n\n\n    var chartObjSel = chartSel.select('.'+hypenate(chart.namespace(),'object-container'))\n    var xAxisObjSel = xAxisSel.select('.'+hypenate(xAxis.namespace(),'object-container'))\n    var yAxisObjSel = yAxisSel.select('.'+hypenate(yAxis.namespace(),'object-container'))\n\n    // xLock = chartSel.node().getBBox().width - chart.spaceX() - chartSel.node().getBBox().x\n    // yLock = chartSel.node().getBBox().height - chart.spaceY()\n    // console.table({'xLock':xLock, \"yLock\":yLock})\n    // bhm.selection().node().getBBox().width - bhm.spaceX()\n\n\n    var chartObjTrans = getTranslation(chartObjSel.attr('transform'))\n    var xAxisObjTrans = getTranslation(xAxisObjSel.attr('transform'))\n    var yAxisObjTrans = getTranslation(yAxisObjSel.attr('transform'))\n\n\n    var x = horizontalQ ? transform.applyX(chartObjTrans[0]) : 0\n    if (horizontalQ) {x = x < -xLock ? (transform.x = 0, -xLock) : (transform.x = 0, Math.min(x, 0)) }\n\n    var y = verticalQ ? transform.applyY(chartObjTrans[1]) : 0\n    if (verticalQ) {y = y < -yLock ? (transform.y = 0, -yLock): (transform.y = 0, Math.min(y, 0))}\n\n    chartObjSel.attr('transform', 'translate('+x+','+y+')')\n    if (horizontalQ) { xAxisObjSel.attr('transform', 'translate('+x+','+0+')') }\n    if (verticalQ) { yAxisObjSel.attr('transform', 'translate('+0+','+y+')') }\n\n    // var lasso = svg.select(\".lasso-container\")\n    // if (!lasso.empty()) {\n    //   lasso.attr('transform', 'translate('+x+','+y+')')\n    // }\n\n  }\n\n  zoom.reset = function() {\n    var horizontalQ, verticalQ\n    if (orient == '2D') {horizontalQ = true; verticalQ = true;}\n    if (orient == 'horizontal') {horizontalQ = true; verticalQ = false;}\n    if (orient == 'vertical') {verticalQ = true; horizontalQ = false;}\n\n    var chartObjSel = chartSel.select('.'+hypenate(chart.namespace(),'object-container'))\n    var xAxisObjSel = xAxisSel.select('.'+hypenate(xAxis.namespace(),'object-container'))\n    var yAxisObjSel = yAxisSel.select('.'+hypenate(yAxis.namespace(),'object-container'))\n    chartObjSel.attr('transform', 'translate('+0+','+0+')')\n    xAxisObjSel.attr('transform', 'translate('+0+','+0+')')\n    yAxisObjSel.attr('transform', 'translate('+0+','+0+')')\n  }\n\n  return zoom\n}\n","import {hypenate, safeSelect, getTranslation} from './helpers';\nimport {log, warn, error, info} from './utils';\n/*******************************************************************************\n\n**                                                                            **\n**                                                                            **\n**                                PLOTZOOM                                    **\n**                                                                            **\n**                                                                            **\n*******************************************************************************/\n/**\n * Creates an plotZoom instance, which can handle drag and scroll events\n * @constructor plotZoom\n * @param {function} chart a function instance of one of the d3sm plots (e.g. bar, boxwhisker, bubbleHeatmap, violin, etc)\n * @param {axis}  xAxis the axis instance responsible for the x axis\n * @param {axis} yAxis the axis instance responsible for the y axis\n * @namespace plotZoom\n * @returns {function} zoom\n */\nexport function multiPlotZoom( chart ) {\n  var\n  /**\n  * The event on which to fire\n  * (see {@link plotZoom#eventType})\n  * @param {string} eventType which event it should handle. Currently supports scroll and wheel\n  * @memberof plotZoom#\n  * @property\n  */\n  eventType,\n  /**\n  * A scaling factor for the wheel \"speed\"\n  * (see {@link plotZoom#wheelSpeed})\n  * @param {number} [wheelSpeed=20] scales the wheel translation by wheelSpeed\n  * @memberof plotZoom#\n  * @property\n  */\n  wheelSpeed = 20,\n  /**\n  * The orientation in which to allow scrolling: 'horizontal', 'vertical', or '2D'\n  * (see {@link plotZoom#orient})\n  * @param {string} [orient=chart.orient() || 'horizontal']\n  * @memberof plotZoom#\n  * @property\n  */\n  orient = (chart.orient == undefined) ? 'horizontal' : chart.orient(),\n  /**\n  * The max distance allowed to scroll in the x direction\n  * (see {@link plotZoom#xLock})\n  * @param {number} [xLock=chart.spaceX()] ideally chart.overflowQ() == true and this value is the\n  * bounding rect across all elements in the chart  minus the space in which to show.\n  * @memberof plotZoom#\n  * @property\n  */\n  xLock=chart.spaceX(),\n  /**\n  * The max distance allowed to scroll in the y direction\n  * (see {@link plotZoom#yLock})\n  * @param {number} [yLock=chart.spaceY()] ideally chart.overflowQ() == true and this value is the\n  * bounding rect across all elements in the chart  minus the space in which to show.\n  * @memberof plotZoom#\n  * @property\n  */\n  yLock=chart.spaceY(),\n\n  chartSel = chart.selection(),\n\n  svg = d3.select(chartSel.thisSVG()),\n\n  xComponents = [],\n  yComponents = []\n\n\n  /**\n   * Gets or sets the event type in which to respond\n   * (see {@link plotZoom#eventType})\n   * @param {string} [_=none] should be 'drag' or 'wheel'\n   * @returns {zoom | string}\n   * @memberof plotZoom\n   * @property\n   * by default plotZoom=undefined\n   */\n  zoom.eventType = function(_) { return arguments.length ? (eventType = _, zoom) : eventType; };\n  /**\n   * Gets or sets the wheel speed in which to scale the wheel scroll transform\n   * (see {@link plotZoom#wheelSpeed})\n   * @param {number} [_=none]\n   * @returns {zoom | number}\n   * @memberof plotZoom\n   * @property\n   * by default wheelSpeed=20\n   */\n  zoom.wheelSpeed = function(_) { return arguments.length ? (wheelSpeed = _, zoom) : wheelSpeed; };\n  /**\n   * Gets or sets the orientation in which items are manipulated\n   * (see {@link plotZoom#orient})\n   * @param {string} [_=none] should be horizontal, vertical, or 2D\n   * @returns {zoom | string}\n   * @memberof plotZoom\n   * @property\n   * by default orient=chart.orient() || 'horizontal'\n   */\n  zoom.orient = function(_) { return arguments.length ? (orient = _, zoom) : orient; };\n\n  /**\n   * Gets or sets the max distance in which to scroll X\n   * (see {@link plotZoom#xLock})\n   * @param {number} [_=none] should be a positive value\n   * @returns {zoom | number}\n   * @memberof plotZoom\n   * @property\n   * by default xLock=chart.spaceX()\n   */\n  zoom.xLock = function(_) { return arguments.length ? (xLock = _, zoom) : xLock; };\n  /**\n   * Gets or sets the max distance in which to scroll Y\n   * (see {@link plotZoom#yLock})\n   * @param {number} [_=none]  should be a positive value\n   * @returns {zoom | number}\n   * @memberof plotZoom\n   * @property\n   * by default yLock=chart.spaceY()\n   */\n  zoom.yLock = function(_) { return arguments.length ? (yLock = _, zoom) : yLock; };\n\n  zoom.xComponents = function(_) { return arguments.length ? (xComponents = _, zoom) : xComponents; };\n  zoom.yComponents = function(_) { return arguments.length ? (yComponents = _, zoom) : yComponents; };\n\n\n  function setLocks() {\n    var chartObjSel = chartSel.select('.'+hypenate(chart.namespace(),'object-container'))\n    var chartObjTrans = getTranslation(chartObjSel.attr('transform'))\n    var cos = chartObjSel.attr('transform', 'translate(0,0)')\n\n    xLock = chartSel.node().getBBox().width - chart.spaceX()// * .9\n    yLock = chartSel.node().getBBox().height - chart.spaceY()// * .9\n    cos.attr('transform', 'translate('+chartObjTrans[0]+','+chartObjTrans[1]+')')\n    log('plotZoom', 'setLocks', {xLock:xLock, yLock:yLock})\n  }\n\n\n  /**\n   * Sets the x and y locks (how far one can scroll)\n   * (see {@link plotZoom#xLock} and {@link plotZoom#yLock})\n   * @function plotZoom.setLocks\n   * @returns {undefined}\n   * @memberof plotZoom\n   * @property\n   */\n  zoom.setLocks = setLocks\n\n  function zoom() {\n    setLocks()\n\n    var\n    xComponentsSel = xComponents.map(function(d, i){return d.selection()}),\n    yComponentsSel = yComponents.map(function(d, i){return d.selection()})\n\n    var horizontalQ, verticalQ\n    if (orient == '2D') {horizontalQ = true; verticalQ = true;}\n    if (orient == 'horizontal') {horizontalQ = true; verticalQ = false;}\n    if (orient == 'vertical') {verticalQ = true; horizontalQ = false;}\n\n    // capture transform event\n    var transform = d3.event.transform\n\n    var chartBox = chartSel.node().getBBox()\n    var xComponentsBox = xComponentsSel.map(function(d, i){return d.node().getBBox()})\n    var yComponentsBox = xComponentsSel.map(function(d, i){return d.node().getBBox()})\n\n\n    var chartWidth = chartBox.width - chartBox.x\n    var chartHeight = chartBox.height - chartBox.y\n\n    // enable wheel event\n    if (eventType == \"wheel\") {\n      var e = d3.event\n      // prevent page scrolling\n      e.preventDefault()\n      // event delta is very very slow, so speed it up with wheel speed\n      var w = d3.event.deltaY * wheelSpeed\n      var shiftQ = d3.event.shiftKey\n\n      // d3 has no way to make custom transform, so make an object and add\n      // the necessary functions to make wheel event compatible with drag events\n      if (orient == '2D') {\n        transform = shiftQ ? {k: 1, x: w, y: 0} : {k: 1, x: 0, y: w}\n      } else {\n        transform = horizontalQ ? {k: 1, x: w, y: 0} : {k: 1, x: 0, y: w}\n      }\n      // * -1 is invert\n      transform.applyX = function(x) { return x * this.k + this.x *-1; }\n      transform.applyY =  function(y) { return y * this.k + this.y *-1; }\n    }\n\n\n\n    var chartObjSel = chartSel.select('.'+hypenate(chart.namespace(),'object-container'))\n    var xComponentObjSel = xComponentsSel.map(function(d, i){\n      return d.select('.'+hypenate(xComponents[i].namespace(),'object-container'))\n    })\n    var yComponentObjSel = yComponentsSel.map(function(d, i){\n      return d.select('.'+hypenate(yComponents[i].namespace(),'object-container'))\n    })\n\n    var chartObjTrans = getTranslation(chartObjSel.attr('transform'))\n    var xComponentsObjTrans = xComponentObjSel.map(function(d, i){\n      return getTranslation(d.attr('transform'))\n    })\n    var yComponentsObjTrans = yComponentObjSel.map(function(d, i){\n      return getTranslation(d.attr('transform'))\n    })\n\n    var x = horizontalQ ? transform.applyX(chartObjTrans[0]) : 0\n    if (horizontalQ) {x = x < -xLock ? (transform.x = 0, -xLock) : (transform.x = 0, Math.min(x, 0)) }\n\n    var y = verticalQ ? transform.applyY(chartObjTrans[1]) : 0\n    if (verticalQ) {y = y < -yLock ? (transform.y = 0, -yLock): (transform.y = 0, Math.min(y, 0))}\n\n    chartObjSel.attr('transform', 'translate('+x+','+y+')')\n    if (horizontalQ) {\n      // xAxisObjSel.attr('transform', 'translate('+x+','+0+')')\n      xComponentObjSel.map(function(d, i){ d.attr('transform', 'translate('+x+','+0+')')  })\n    }\n    if (verticalQ) {\n      // yAxisObjSel.attr('transform', 'translate('+0+','+y+')')\n      yComponentObjSel.map(function(d, i){ d.attr('transform', 'translate('+0+','+y+')')  })\n\n    }\n\n\n  }\n\n  zoom.reset = function() {\n    var horizontalQ, verticalQ\n    if (orient == '2D') {horizontalQ = true; verticalQ = true;}\n    if (orient == 'horizontal') {horizontalQ = true; verticalQ = false;}\n    if (orient == 'vertical') {verticalQ = true; horizontalQ = false;}\n\n    var chartObjSel = chartSel.select('.'+hypenate(chart.namespace(),'object-container'))\n    var xAxisObjSel = xAxisSel.select('.'+hypenate(xAxis.namespace(),'object-container'))\n    var yAxisObjSel = yAxisSel.select('.'+hypenate(yAxis.namespace(),'object-container'))\n    chartObjSel.attr('transform', 'translate('+0+','+0+')')\n    xAxisObjSel.attr('transform', 'translate('+0+','+0+')')\n    yAxisObjSel.attr('transform', 'translate('+0+','+0+')')\n  }\n\n  return zoom\n}\n","import {hypenate, safeSelect, modifyHexidecimalColorLuminance, extractViolinValues, quartiles} from './helpers';\nimport {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer} from './utils';\nimport {unique, hasQ, flatten, whichBin} from './array-functions';\nimport {groupingSpacer} from './grouping-spacer';\nimport {colorFunction as CF} from './color-function';\nimport {tooltip as TTip} from './tooltip';\n\n/*******************************************************************************\n**                                                                            **\n**                                                                            **\n**                                 VIOLIN                                     **\n**                                                                            **\n**                                                                            **\n*******************************************************************************/\n\n/**\n * Creates a violin\n *\n * {@link https://sumneuron.gitlab.io/d3sm/demos/basic-violins/index.html Demo}\n * @constructor violin\n * @param {d3.selection} selection\n * @namespace violin\n * @returns {function} violin\n */\nexport function violin( selection ) {\n  var\n  /**\n  * Data to plot. Assumed to be a object, where each key corresponds to a violin\n  * (see {@link violin#data})\n  * @param {Object} [data=undefined]\n  * @memberof violin#\n  * @property\n  */\n  data,\n  /**\n  * Which direction to render the bars in\n  * (see {@link violin#orient})\n  * @param {number} [orient='horizontal']\n  * @memberof violin#\n  * @property\n  */\n  orient='horizontal',\n  /**\n  * Amount of horizontal space (in pixels) avaible to render the violin in\n  * (see {@link violin#spaceX})\n  * @param {number} [spaceX=undefined]\n  * @memberof violin#\n  * @property\n  */\n  spaceX,\n  /**\n  * Amount of vertical space (in pixels) avaible to render the violin in\n  * (see {@link violin.spaceY})\n  * @param {number} [spaceY=undefined]\n  * @memberof violin#\n  * @property\n  */\n  spaceY,\n  /**\n  * Whether or not to allow violin to render elements pass the main spatial dimension\n  * given the orientation (see {@link violin#orient}), where {@link violin#orient}=\"horizontal\"\n  * the main dimension is {@link violin#spaceX} and where {@link violin#orient}=\"vertical\"\n  * the main dimension is {@link violin#spaceY}\n  * @param {boolean} [overflowQ=false]\n  * @memberof violin#\n  * @property\n  */\n  overflowQ = true,\n  /**\n  * Whether or not to display points inside the points\n  * @param {boolean} [pointsQ=false]\n  * @memberof violin#\n  * @property\n  */\n  pointsQ = true,\n  /**\n  * An array - putatively of other arrays - depicting how bars should be arranged\n  * @param {Array[]} [grouping=undefined]\n  * @memberof violin#\n  * @property\n  */\n  grouping,\n  /**\n  * How to get the value of the violin\n  * @param {function} [valueExtractor=function(key, index) { return data[key] }]\n  * @memberof violin#\n  * @property\n  */\n  valueExtractor = function(key, index) {return data[key] },\n  /**\n  * How to sort the bars - if {@link violin#grouping} is not provided.\n  * @param {function} [sortingFunction=function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}]\n  * @memberof violin#\n  * @property\n  */\n  sortingFunction = function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])},\n\n  /**\n  * The scale for which violin values should be transformed by\n  * @param {d3.scale} [scale=d3.scaleLinear]\n  * @memberof violin#\n  * @property\n  */\n  scale = d3.scaleLinear(),\n  /**\n  * The padding for the domain of the scale (see {@link violin#scale})\n  * @param {number} [domainPadding=0.5]\n  * @memberof violin#\n  * @property\n  */\n  domainPadding = 0.5,\n  /**\n  * Default space for the spacer (percentage) of main dimension given the orientation\n  * (see {@link violin#orient}), where {@link violin#orient}=\"horizontal\"\n  * the main dimension is {@link violin#spaceX} and where {@link violin#orient}=\"vertical\"\n  * the main dimension is {@link violin#spaceY} between bars\n  * @param {number} [objectSpacer=0.05]\n  * @memberof violin#\n  * @property\n  */\n  objectSpacer = 0.05,\n  /**\n  * The minimum size that an object can be\n  * @param {number} [minObjectSize=50]\n  * @memberof violin#\n  * @property\n  */\n  minObjectSize = 50,\n  /**\n  * The maximum size that an object can be\n  * @param {number} [maxObjectSize=100]\n  * @memberof violin#\n  * @property\n  */\n  maxObjectSize = 100,\n\n  /**\n  * The stroke width of the bars\n  * @param {number} [barStrokeWidth=2]\n  * @memberof violin#\n  * @property\n  */\n  objectStrokeWidth = 2,\n  /**\n  * Instance of ColorFunction\n  * @param {function} [colorFunction = colorFunction()]\n  * @memberof violin#\n  * @property\n  */\n  colorFunction = CF(),\n  /**\n  * Instance of ColorFunction modified by a scale for the points\n  * @param {function} [pointColorFunc = colorFunction()]\n  * @memberof violin#\n  * @property\n  */\n  pointColorFunc = function (d, type, base, min, max) {\n    var minMaxHexScale = d3.scaleLinear().domain([min, max]).range([-0.25, 0.05])\n    var scaledColor = modifyHexidecimalColorLuminance(base.replace('#', ''), minMaxHexScale(d))\n    var mod = type == \"stroke\" ? 0 : 0.25\n    return modifyHexidecimalColorLuminance(scaledColor.replace('#', ''), mod)\n  },\n\n  /**\n  * The radius of a point\n  * @param {number} [pointRadius=3]\n  * @memberof violin#\n  * @property\n  */\n  pointRadius = 3,\n  /**\n  * The stroke width of the oints\n  * @param {number} [pointStrokeWidth=2]\n  * @memberof violin#\n  * @property\n  */\n  pointStrokeWidth = 2,\n\n  /**\n  * Color of the background\n  * @param {string} [backgroundFill=\"transparent\"]\n  * @memberof violin#\n  * @property\n  */\n  backgroundFill = 'transparent',\n  /**\n  * Namespace for all items made by this instance of violin\n  * @param {string} [namespace=\"d3sm-violin\"]\n  * @memberof violin#\n  * @property\n  */\n  namespace = 'd3sm-violin',\n  /**\n  * Class name for violin container (<g> element)\n  * @param {string} [objectClass=\"violin\"]\n  * @memberof violin#\n  * @property\n  */\n  objectClass = 'violin',\n  /**\n  * Duration of all transitions of this element\n  * @param {number} [transitionDuration=1000]\n  * @memberof violin#\n  * @property\n  */\n  transitionDuration = 1000,\n  /**\n  * Easing function for transitions\n  * @param {d3.ease} [easeFunc=d3.easeExp]\n  * @memberof violin#\n  * @property\n  */\n  easeFunc = d3.easeExp,\n\n  /**\n  * The key containing the quartiles\n  * @param {string} [quartilesKey=undefined]\n  * @memberof violin#\n  * @property\n  */\n  quartilesKey = \"quartiles\",\n  /**\n  * The keys corresponding to each quartile\n  * @param {string[]} [quartileKeys=[\"Q0\", \"Q1\", \"Q2\", \"Q3\", \"Q4\"]]\n  * @memberof violin#\n  * @property\n  */\n  quartileKeys = [\"Q0\", \"Q1\", \"Q2\", \"Q3\", \"Q4\"],\n\n  /**\n  * The keys of the bars\n  * @param {string[]} [violinKeys=undefined]\n  * @memberof violin#\n  * @property\n  */\n  violinKeys,\n  /**\n  * The values of the bars\n  * @param {number[]} [violinValues=undefined]\n  * @memberof violin#\n  * @property\n  */\n  violinValues,\n  /**\n  * The objectSize (actual width) used by the bars\n  * @param {number} [objectSize=undefined]\n  * @memberof violin#\n  * @property\n  */\n  objectSize,\n  /**\n  * The spacerSize (actual width) used by the spacers between the bars\n  * @param {number} [spacerSize=undefined]\n  * @memberof violin#\n  * @property\n  */\n  spacerSize,\n\n  /**\n  * Instance of Tooltip\n  * @param {function} [tooltip=tooltip()]\n  * @memberof violin#\n  * @property\n  */\n  tooltip = TTip().keys([quartileKeys[4], quartileKeys[3], quartileKeys[2], quartileKeys[1], quartileKeys[0]]),\n  pointsTooltip = TTip(),\n  // pointKeyExtractor = function(violinKey, violinData, violinValues) {return d3.keys(violinValues[violinKey].values)},\n  // pointValueExtractor = function(pointKey, violinKey, violinData, violinValues) {return violinValues[violinKey].values[pointKey]},\n\n\n  /**\n  * Function which given the key of the violin and that key's associated value\n  * returns the object consiting of the points of the violin\n  * (see {@link violin#violinPointsExtractor})\n  * @param {Object} [violinPointsExtractor=function(violinKey, violinData) { return violinData.points }]\n  * @memberof violin#\n  * @property\n  */\n  violinPointsExtractor = function (violinKey, violinData) {return violinData.points },\n  /**\n  * Function which given the key of the current point and the object of points for the\n  * violin, returns the numerical value of the point\n  * (see {@link violin#violinPointValueExtractor})\n  * @param {Object} [violinPointValueExtractor=function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value }]\n  * @memberof violin#\n  * @property\n  */\n  violinPointValueExtractor = function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value }\n\n\n  /**\n   * Gets or sets the violinPointsExtractor\n   * @param {function} [_=none]\n   * @returns {violin | function}\n   * @memberof violin\n   * @property\n   * by default violinPointsExtractor = function(violinKey, violinData) { return violinData.points }\n   */\n  violin.violinPointsExtractor = function(_) { return arguments.length ? (violinPointsExtractor = _, violin) : violinPointsExtractor; };\n  /**\n   * Gets or sets the violinPointValueExtractor\n   * @param {function} [_=none]\n   * @returns {violin | function}\n   * @memberof violin\n   * @property\n   * by default violinPointsExtractor = function(pointKey, violinKey, violinData, violinValues) {return violinValues[violinKey].values[pointKey]}\n   */\n  violin.violinPointValueExtractor = function(_) { return arguments.length ? (violinPointValueExtractor = _, violin) : violinPointValueExtractor; };\n\n\n  /**\n   * Gets or sets the selection in which items are manipulated\n   * @param {d3.selection} [_=none]\n   * @returns {violin | d3.selection}\n   * @memberof violin\n   * @property\n   * by default selection = selection\n   */\n  violin.selection = function(_) { return arguments.length ? (selection = _, violin) : selection; };\n  /**\n   * Gets or sets the data\n   * (see {@link violin#data})\n   * @param {number} [_=none]\n   * @returns {violin | object}\n   * @memberof violin\n   * @property\n   */\n  violin.data = function(_) { return arguments.length ? (data = _, violin) : data; };\n  /**\n   * Gets or sets the orient of the boxes\n   * (see {@link violin#orient})\n   * @param {number} [_=none]\n   * @returns {violin | object}\n   * @memberof violin\n   * @property\n   */\n  violin.orient = function(_) { return arguments.length ? (orient = _, violin) : orient; };\n  /**\n   * Gets or sets the amount of horizontal space in which items are manipulated\n   * (see {@link violin#spaceX})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {violin | number}\n   * @memberof violin\n   * @property\n   * by default spaceX = undefined\n   */\n  violin.spaceX = function(_) { return arguments.length ? (spaceX = _, violin) : spaceX; };\n  /**\n   * Gets or sets the amount of vertical space in which items are manipulated\n   * (see {@link violin#spaceY})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {violin | number}\n   * @memberof violin\n   * @property\n   * by default spaceY = undefined\n   */\n  violin.spaceY = function(_) { return arguments.length ? (spaceY = _, violin) : spaceY; };\n\n\n  /**\n   * Gets / sets whether or not violin is allowed to go beyond specified dimensions\n   * (see {@link violin#overflowQ})\n   * @param {boolean} [_=none]\n   * @returns {violin | boolean}\n   * @memberof violin\n   * @property\n   * by default overflowQ = false\n   */\n  violin.overflowQ = function(_) { return arguments.length ? (overflowQ = _, violin) : overflowQ; };\n  /**\n   * Gets / sets whether or not to plot points with the violins\n   * (see {@link violin#pointsQ})\n   * @param {boolean} [_=none]\n   * @returns {violin | boolean}\n   * @memberof violin\n   * @property\n   * by default pointsQ = false\n   */\n  violin.pointsQ = function(_) { return arguments.length ? (pointsQ = _, violin) : pointsQ; };\n\n\n  /**\n   * Gets / sets the grouping of the boxes\n   * (see {@link violin#grouping})\n   * @param {Array[]} [_=none]\n   * @returns {violin | Array[]}\n   * @memberof violin\n   * @property\n   * by default grouping = undefined\n   */\n  violin.grouping = function(_) { return arguments.length ? (grouping = _, violin) : grouping; };\n  /**\n   * Gets / sets the valueExtractor\n   * (see {@link violin#valueExtractor})\n   * @param {function} [_=none]\n   * @returns {violin | function}\n   * @memberof violin\n   * @property\n   */\n  violin.valueExtractor = function(_) { return arguments.length ? (valueExtractor = _, violin) : valueExtractor; };\n  /**\n   * Gets / sets the sortingFunction\n   * (see {@link violin#sortingFunction})\n   * @param {function} [_=none]\n   * @returns {violin | function}\n   * @memberof violin\n   * @property\n   */\n  violin.sortingFunction = function(_) { return arguments.length ? (sortingFunction = _, violin) : sortingFunction; };\n\n  /**\n   * Gets / sets the scale for which the violin values should be transformed by\n   * (see {@link violin#scale})\n   * @param {d3.scale} [_=none]\n   * @returns {violin | d3.scale}\n   * @memberof violin\n   * @property\n   * by default scale = d3.scaleLinear()\n   */\n  violin.scale = function(_) { return arguments.length ? (scale = _, violin) : scale; };\n  /**\n   * Gets / sets the padding for the domain of the scale\n   * (see {@link violin#domainPadding})\n   * @param {number} [_=none]\n   * @returns {violin | number}\n   * @memberof violin\n   * @property\n   * by default domainPadding = 0.5\n   */\n  violin.domainPadding = function(_) { return arguments.length ? (domainPadding = _, violin) : domainPadding; };\n\n\n  /**\n   * Gets / sets objectSpacer\n   * (see {@link violin#objectSpacer})\n   * @param {number} [_=none]\n   * @returns {violin | number}\n   * @memberof violin\n   * @property\n   * by default objectSpacer = 0.05\n   */\n  violin.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, violin) : objectSpacer; };\n  /**\n   * Gets / sets the minObjectSize\n   * (see {@link violin#minObjectSize})\n   * @param {number} [_=none]\n   * @returns {violin | number}\n   * @memberof violin\n   * @property\n   * by default minObjectSize = 15\n   */\n  violin.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, violin) : minObjectSize; };\n  /**\n   * Gets / sets the maxObjectSize\n   * (see {@link violin#maxObjectSize})\n   * @param {number} [_=none]\n   * @returns {violin | number}\n   * @memberof violin\n   * @property\n   * by default maxObjectSize = 50\n   */\n  violin.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, violin) : maxObjectSize; };\n\n  /**\n   * Gets / sets the objectStrokeWidth\n   * (see {@link violin#objectStrokeWidth})\n   * @param {number} [_=none]\n   * @returns {violin | number}\n   * @memberof violin\n   * @property\n   * by default objectStrokeWidth = 2\n   */\n  violin.objectStrokeWidth = function(_) { return arguments.length ? (objectStrokeWidth = _, violin) : objectStrokeWidth; };\n\n\n  /**\n   * Gets / sets the colorFunction\n   * (see {@link violin#colorFunction})\n   * @param {colorFunction} [_=none]\n   * @returns {violin | colorFunction}\n   * @memberof violin\n   * @property\n   * by default colorFunction = colorFunction()\n   */\n  violin.colorFunction = function(_) { return arguments.length ? (colorFunction = _, violin) : colorFunction; };\n  /**\n   * Gets / sets the colorFunction\n   * (see {@link violin#colorFunction})\n   * @param {colorFunction} [_=none]\n   * @returns {violin | colorFunction}\n   * @memberof violin\n   * @property\n   * by default colorFunction = colorFunction()\n   */\n  violin.pointColorFunc = function(_) { return arguments.length ? (pointColorFunc = _, violin) : pointColorFunc; };\n\n\n  /**\n   * Gets / sets the pointRadius\n   * (see {@link violin#pointRadius})\n   * @param {number} [_=none]\n   * @returns {violin | number}\n   * @memberof violin\n   * @property\n   * by default pointRadius = 2\n   */\n  violin.pointRadius = function(_) { return arguments.length ? (pointRadius = _, violin) : pointRadius; };\n  /**\n   * Gets / sets the pointStrokeWidth\n   * (see {@link violin#pointStrokeWidth})\n   * @param {number} [_=none]\n   * @returns {violin | number}\n   * @memberof violin\n   * @property\n   * by default pointStrokeWidth = 2\n   */\n  violin.pointStrokeWidth = function(_) { return arguments.length ? (pointStrokeWidth = _, violin) : pointStrokeWidth; };\n\n\n  /**\n   * Gets / sets the backgroundFill\n   * (see {@link violin#backgroundFill})\n   * @param {string} [_=none]\n   * @returns {violin | string}\n   * @memberof violin\n   * @property\n   * by default backgroundFill = 'transparent'\n   */\n  violin.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, violin) : backgroundFill; };\n  /**\n   * Gets / sets the namespace\n   * (see {@link violin#namespace})\n   * @param {string} [_=none]\n   * @returns {violin | string}\n   * @memberof violin\n   * @property\n   * by default namespace = 'd3sm-violin'\n   */\n  violin.namespace = function(_) { return arguments.length ? (namespace = _, violin) : namespace; };\n  /**\n   * Gets / sets the objectClass\n   * (see {@link violin#objectClass})\n   * @param {string} [_=none]\n   * @returns {violin | string}\n   * @memberof violin\n   * @property\n   * by default objectClass = 'tick-group'\n   */\n  violin.objectClass = function(_) { return arguments.length ? (objectClass = _, violin) : objectClass; };\n\n\n  /**\n   * Gets / sets the transitionDuration\n   * (see {@link violin#transitionDuration})\n   * @param {number} [_=none]\n   * @returns {violin | number}\n   * @memberof violin\n   * @property\n   * by default transitionDuration = 1000\n   */\n  violin.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, violin) : transitionDuration; };\n  /**\n   * Gets / sets the easeFunc\n   * (see {@link violin#easeFunc})\n   * @param {d3.ease} [_=none]\n   * @returns {violin | d3.ease}\n   * @memberof violin\n   * @property\n   * by default easeFunc = d3.easeExp\n   */\n  violin.easeFunc = function(_) { return arguments.length ? (easeFunc = _, violin) : easeFunc; };\n\n\n  /**\n   * Gets / sets the quartileKey\n   * (see {@link violin#quartileKey})\n   * @param {string} [_=none]\n   * @returns {violin | string}\n   * @memberof violin\n   * @property\n   * by default quartileKey = \"quartiles\"\n   */\n  violin.quartileKey = function(_) { return arguments.length ? (quartileKey = _, violin) : quartileKey; };\n  /**\n   * Gets / sets the quartileKeys\n   * (see {@link violin#quartileKeys})\n   * @param {string[]} [_=none]\n   * @returns {violin | string[]}\n   * @memberof violin\n   * @property\n   * by default quartileKeys = [\"Q0\",\"Q1\",\"Q2\",\"Q3\",\"Q4\"]\n   */\n  violin.quartileKeys = function(_) { return arguments.length ? (quartileKeys = _, violin) : quartileKeys; };\n\n\n  /**\n   * Gets / sets the violinKeys\n   * (see {@link violin#violinKeys})\n   * @param {string[]} [_=none]\n   * @returns {violin | string[]}\n   * @memberof violin\n   * @property\n   * by default violinKeys = undefined\n   */\n  violin.violinKeys = function(_) { return arguments.length ? (violinKeys = _, violin) : violinKeys; };\n  /**\n   * Gets / sets the violinValues\n   * (see {@link violin#violinValues})\n   * @param {Object[]} [_=none]\n   * @returns {violin | Object[]}\n   * @memberof violin\n   * @property\n   * by default violinValues = undefined\n   */\n  violin.violinValues = function(_) { return arguments.length ? (violinValues = _, violin) : violinValues; };\n\n  /**\n   * Gets / sets the objectSize\n   * (see {@link violin#objectSize})\n   * @param {number} [_=none]\n   * @returns {violin | number}\n   * @memberof violin\n   * @property\n   * by default objectSize = undefined\n   */\n  violin.objectSize = function(_) { return arguments.length ? (objectSize = _, violin) : objectSize; };\n  /**\n   * Gets / sets the spacerSize\n   * (see {@link violin#spacerSize})\n   * @param {number} [_=none]\n   * @returns {violin | number}\n   * @memberof violin\n   * @property\n   * by default spacerSize = undefined\n   */\n  violin.spacerSize = function(_) { return arguments.length ? (spacerSize = _, violin) : spacerSize; };\n  /**\n   * Gets / sets the tooltip\n   * (see {@link violin#tooltip})\n   * @param {tooltip} [_=none]\n   * @returns {violin | tooltip}\n   * @memberof violin\n   * @property\n   * by default tooltip = tooltip()\n   */\n  violin.tooltip = function(_) { return arguments.length ? (tooltip = _, violin) : tooltip; };\n  /**\n   * Gets / sets the pointsTooltip\n   * (see {@link violin#pointsTooltip})\n   * @param {tooltip} [_=none]\n   * @returns {violin | tooltip}\n   * @memberof violin\n   * @property\n   * by default pointsTooltip = tooltip()\n   */\n  violin.pointsTooltip = function(_) { return arguments.length ? (pointsTooltip = _, violin) : pointsTooltip; };\n\n\n\n\n\n  function violin () {\n    // for convenience in handling orientation specific values\n    var horizontalQ = (orient == 'horizontal') ? true : false\n    var verticalQ = !horizontalQ\n\n    // background cliping rectangle\n    var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}\n    var container = setupContainer( selection, namespace, bgcpRect, backgroundFill );\n\n    // if grouping is undefined sort violinKeys by sortingFunction\n    var ordered = (grouping == undefined) ? d3.keys(data).sort(sortingFunction) : grouping\n\n    // console.log(ordered)\n\n    violinKeys = flatten(ordered)\n\n\n    var calcValues = neededViolinValues()\n    .horizontalQ(horizontalQ)\n    .quartileKeys(quartileKeys)\n    .violinPointsExtractor(violinPointsExtractor)\n    .violinPointValueExtractor(violinPointValueExtractor)\n\n\n\n    // augment valus\n    violinKeys.map(function(vk, i){ calcValues(vk, data) })\n\n    var numberOfObjects = violinKeys.length\n\n    var min = [].concat(...violinKeys.map(function(k, i){return data[k].quartiles[quartileKeys[0]]}))\n    var max = [].concat(...violinKeys.map(function(k, i){return data[k].quartiles[quartileKeys[quartileKeys.length - 1]]}))\n    var extent = [Math.min(...min) - domainPadding, Math.max(...max) + domainPadding]\n    // console.log(extent, violinValues, ordered)\n\n    // set the scale\n    scale.domain(extent).range(horizontalQ ? [0,spaceY] : [0, spaceX])\n    var space = horizontalQ ? spaceX : spaceY\n    // calculate object size\n    objectSize = calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ)\n    // calculate spacer size if needed\n    spacerSize = calculateWidthOfSpacer(ordered, space, objectSize, numberOfObjects, objectSpacer, overflowQ)\n    // make the nested groups\n    var spacerFunction = groupingSpacer()\n    .horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects)\n    .objectClass(objectClass).objectSize(objectSize).spacerSize(spacerSize)\n    .transitionDuration(transitionDuration).easeFunc(easeFunc)\n    .namespace(namespace)\n\n    // move stuff\n    spacerFunction(container, ordered, 0)\n    // console.log(violinKeys, ordered, container.selectAll('g:not(.to-remove).'+objectClass).nodes())\n\n    // for color function\n    var parentIndexArray = []\n    container.selectAll('g:not(.to-remove).'+objectClass)\n    .each(function(d, i){if (hasQ(violinKeys, d)){ parentIndexArray.push(Number(d3.select(this).attr('parent-index')))}})\n\n    // update color function\n    colorFunction = colorFunction.colorBy() == 'index'\n    ? colorFunction.dataExtent([0, Math.max(...parentIndexArray)])\n    : colorFunction.dataExtent(extent)\n\n    /* violiin specific needs */\n\n\n    var frequencyMax = Math.max(...[].concat(...violinKeys.map(function(k, i){return d3.max(data[k].frequencies)})))\n    var vScale = d3.scaleLinear().domain([0, frequencyMax]).range([0, objectSize / 2])\n\n    var lArea = d3.line()\n    .x(function(d, i){ return horizontalQ ? -vScale(d.x) : scale(d.x)})\n    .y(function(d, i){ return horizontalQ ? scale(extent[1]) - scale(d.y) : -vScale(d.y)})\n    .curve(d3.curveBasis)\n    var rArea = d3.line()\n    .x(function(d, i){ return horizontalQ ? vScale(d.x) : scale(d.x)})\n    .y(function(d, i){ return horizontalQ ? scale(extent[1]) - scale(d.y) : vScale(d.y)})\n    .curve(d3.curveBasis)\n\n\n\n\n\n\n    container.selectAll('g:not(.to-remove).'+objectClass).each(function(key, i){\n      var t = d3.select(this),\n      currentData = data[key]\n      // needed because bug in selecting .to-remove\n      if (!hasQ(violinKeys, key)) {return}\n      var\n      i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'),\n      fillColor = colorFunction(key, currentData, i, 'fill'), // prevent duplicate computation\n      strokeColor = colorFunction(key, currentData, i, 'stroke'),\n      area = safeSelect(t, 'g', 'area'),\n      la = safeSelect(area, 'path', 'left'),\n      ra = safeSelect(area, 'path', 'right'),\n      quarts = safeSelect(t, 'g', 'quarts'),\n      lq3 = safeSelect(quarts, 'line', 'q3'),\n      lq1 = safeSelect(quarts, 'line', 'q1'),\n      q3 = currentData.quartiles[quartileKeys[3]],\n      q2 = currentData.quartiles[quartileKeys[2]],\n      q1 = currentData.quartiles[quartileKeys[1]]\n\n      t.attr('transform', horizontalQ ? 'translate('+objectSize / 2+',0)' : 'translate(0,'+objectSize / 2+')'  )\n      // draw curve\n      la.transition().duration(transitionDuration).attr('d', function(dd, ii){ return lArea(currentData.contour)})\n      .attr('fill', fillColor)\n      .attr('stroke', strokeColor)\n      .attr('stroke-width', objectStrokeWidth)\n\n      ra.transition().duration(transitionDuration).attr('d', function(dd, ii){ return rArea(currentData.contour)})\n      .attr('fill', fillColor)\n      .attr('stroke', strokeColor)\n      .attr('stroke-width', objectStrokeWidth)\n\n      area.node().addEventListener('mouseover', function(dd, ii){\n        container.selectAll('g.'+objectClass).style('opacity', 0.2)\n        t.style('opacity', 1)\n        la.attr('stroke-width',objectStrokeWidth*2)\n        ra.attr('stroke-width',objectStrokeWidth*2)\n      })\n      area.node().addEventListener('mouseout', function(dd, ii){\n        container.selectAll('g.'+objectClass).style('opacity', 1)\n        la.attr('stroke-width',objectStrokeWidth)\n        ra.attr('stroke-width',objectStrokeWidth)\n      })\n\n      if (pointsQ) {\n        var ptsContainer = safeSelect(t, 'g', 'points')\n        var pts = ptsContainer.selectAll('.point').data(currentData.pointKeys)\n        pts.on('mouseover', null)\n\n\n        var ptsExit = pts.exit().transition().ease(easeFunc).duration(transitionDuration)\n        .attr('r', 0)\n        .attr('cy', horizontalQ ? scale(extent[1]) - scale(q2) : vScale(0))\n        .attr('cx', horizontalQ ? vScale(0) : scale(q2)).remove()\n\n        var ptsEnter = pts.enter().append('circle').attr('class', 'point').attr('r', 0)\n        .attr('cx', horizontalQ ? 0 : scale(q2))\n        .attr('cy', horizontalQ ? scale(q2) : 0)\n\n        pts = pts.merge(ptsEnter)\n\n        // console.log(pointsTooltip.header())\n\n        var pTTips = TTip().selection(pts).data(violinPointsExtractor(key, currentData))\n        .header(pointsTooltip.header())\n        .keys(pointsTooltip.keys())\n        .values(pointsTooltip.values())\n\n        pTTips()\n\n        var pMin = d3.min(currentData.pointValues), pMax = d3.max(currentData.pointValues)\n\n\n\n        pts.transition().duration(transitionDuration).ease(easeFunc).attr('r', pointRadius)\n        .attr('cy', function(pointKey, ii){\n          var dd = currentData.pointValues[ii]\n          if (horizontalQ) { return scale(extent[1]) - scale(dd) }\n          var j = whichBin(currentData.binned, dd)\n          var r = Math.random()\n          var n = vScale(r * currentData.frequencies[j] * 0.5)\n          var k = Math.random() > 0.5 ? n : -n\n          return k\n        })\n        .attr('cx', function(pointKey, ii){\n          var dd = currentData.pointValues[ii]\n          if (horizontalQ) {\n            var j = whichBin(currentData.binned, dd)\n            var r = Math.random()\n            var n = vScale(r * currentData.frequencies[j] * 0.5)\n            var k = Math.random() > 0.5 ? n : -n\n            return k\n          }\n          return scale(dd)\n        })\n        .attr('stroke', function(dd, ii) { var dd = currentData.pointValues[ii]; return pointColorFunc(dd, 'stroke', strokeColor, pMin, pMax) })\n        .attr('fill'  , function(dd, ii) { var dd = currentData.pointValues[ii]; return pointColorFunc(dd, 'fill'  , strokeColor, pMin, pMax) })\n        .attr('stroke-width', pointStrokeWidth)\n\n        ptsContainer.selectAll('circle.point').on('mouseover', function(dd, ii){\n          container.selectAll('g.'+objectClass).style('opacity', 0.2)\n          t.style('opacity', 1)\n          la.attr('stroke-width',objectStrokeWidth*2)\n          ra.attr('stroke-width',objectStrokeWidth*2)\n\n          container.selectAll('.point').style('opacity', 0.2)\n          d3.select(this).style('opacity', 1).attr('r', pointRadius * 2).attr('stroke-width',pointStrokeWidth*2)\n        })\n        ptsContainer.selectAll('circle.point').on('mouseout', function(dd, ii){\n          var e = document.createEvent('SVGEvents')\n          e.initEvent('mouseout',true,true);\n          area.node().dispatchEvent(e)\n\n          container.selectAll('.point').style('opacity', 1)\n          d3.select(this).attr('stroke-width', pointStrokeWidth).attr('r', pointRadius)\n        })\n      }\n      else {\n        cV.selectAll('.point')\n        .transition().duration(transitionDuration).ease(easeFunc)\n        .attr('r', 0)\n        .attr('cy', horizontalQ ? scale(extent[1]) - scale(q2) : vScale(0))\n        .attr('cx', horizontalQ ? vScale(0) : scale(q2))\n        .remove()\n      }\n\n\n    })\n\n\n    tooltip.selection(container.selectAll('g:not(.to-remove).'+objectClass + ' .area'))\n    if (tooltip.data() == undefined) {tooltip.data(data)}\n    tooltip()\n    if (tooltip.values() == undefined) {\n      tooltip.values([\n        function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] },\n        function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] },\n        function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] },\n        function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] },\n        function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] }\n      ])\n\n    }\n\n  }\n\n  return violin\n}\n\n\n\n\n\n/**\n* Produces the function which manipulates the violin data to have values needed\n* for rendering the violins as svg.\n* @returns {function} calculateViolinValues\n* @namespace neededViolinValues\n*/\nfunction neededViolinValues() {\n  var\n  /**\n  * Whether or not the orientation of the violins are horizontal\n  * (see {@link violin#orient})\n  * @param {Object} [horizontalQ=true]\n  * @memberof neededViolinValues#\n  * @property\n  */\n  horizontalQ = true,\n  /**\n  * Keys to be put into the quartiles if they need to be calculated.\n  * (see {@link violin#quartileKeys})\n  * @param {Object} [quartileKeys=['Q0', 'Q1', 'Q2', 'Q3', 'Q4']]\n  * @memberof neededViolinValues#\n  * @property\n  */\n  quartileKeys = ['Q0', 'Q1', 'Q2', 'Q3', 'Q4'],\n  /**\n  * Function which given the key of the violin and that key's associated value\n  * returns the object consiting of the points of the violin\n  * (see {@link violin#violinPointsExtractor})\n  * @param {Object} [violinPointsExtractor=function(violinKey, violinData) { return violinData.points }]\n  * @memberof neededViolinValues#\n  * @property\n  */\n  violinPointsExtractor,\n  /**\n  * Function which given the key of the current point and the object of points for the\n  * violin, returns the numerical value of the point\n  * (see {@link violin#violinPointValueExtractor})\n  * @param {Object} [violinPointValueExtractor=function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value }]\n  * @memberof neededViolinValues#\n  * @property\n  */\n  violinPointValueExtractor\n\n\n  /**\n   * Gets / sets the horizontalQ\n   * (see {@link violin#orient})\n   * @param {boolean} [_=none]\n   * @returns {calculateViolinValues | boolean}\n   * @memberof calculateViolinValues\n   * @property\n   *\n   * by default horizontalQ = true\n   */\n  calculateViolinValues.horizontalQ = function(_) { return arguments.length ? (horizontalQ=_, calculateViolinValues) : horizontalQ }\n  /**\n   * Gets / sets the quartileKeys\n   * (see {@link violin#quartileKeys})\n   * @param {string[]} [_=none]\n   * @returns {calculateViolinValues | string[]}\n   * @memberof calculateViolinValues\n   * @property\n   *\n   * by default quartileKeys = [\"Q0\",\"Q1\",\"Q2\",\"Q3\",\"Q4\"]\n   */\n  calculateViolinValues.quartileKeys = function(_) { return arguments.length ? (quartileKeys=_, calculateViolinValues) : quartileKeys }\n  /**\n   * Gets / sets the violinPointsExtractor\n   * (see {@link violin#violinPointsExtractor})\n   * @param {function} [_=none]\n   * @returns {calculateViolinValues | function}\n   * @memberof calculateViolinValues\n   * @property\n   *\n   * by default violinPointsExtractor = function(violinKey, violinData) { return violinData.points }\n   */\n  calculateViolinValues.violinPointsExtractor = function(_) { return arguments.length ? (violinPointsExtractor=_, calculateViolinValues) : violinPointsExtractor }\n  /**\n   * Gets / sets the violinPointValueExtractor\n   * (see {@link violin#violinPointValueExtractor})\n   * @param {function} [_=none]\n   * @returns {calculateViolinValues | function}\n   * @memberof calculateViolinValues\n   * @property\n   *\n   * by default violinPointValueExtractor = function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value }\n   */\n  calculateViolinValues.violinPointValueExtractor = function(_) { return arguments.length ? (violinPointValueExtractor=_, calculateViolinValues) : violinPointValueExtractor }\n\n\n  /**\n  * The function produced by neededViolinValues.\n  *\n  * Adds the data need to render the violin as an svg\n  * @param {string} violinKey the key of the current violin\n  * @param {object} data the object consisting of violinKey - values (violin data) pairs\n  * @returns {none} this function manipulates the passed data object adding the following keys\n  *\n  * data[violinKey].binned // the binned values of the points\n  *\n  * data[violinKey].frequencies // the list consisting of the length of each bin\n  *\n  * data[violinKey].contour // the points depicting the contour of 1/2 of the violin\n  *\n  * data[violinKey].quartiles // the quartiles of the points' values\n  *\n  * data[violinKey].pointKeys // the keys associated with the points\n  *\n  * data[violinKey].pointValues // the numerical values of the points\n  *\n  * @memberof neededViolinValues#\n  * @property\n  */\n  function calculateViolinValues(violinKey, data) {\n    // data for the current violin\n    var violinData = data[violinKey];\n    // the object of points\n    var violinPoints = violinPointsExtractor(violinKey, violinData);\n    //\n    var violinPointsKeys = d3.keys(violinPoints);\n    // the numerical values of those points\n    var violinPointsValues = violinPointsKeys.map(function(pk, i){return violinPointValueExtractor(pk, violinPoints)})\n\n    // quartiles of those points\n    var pointQuartiles = quartiles(violinPointsValues, quartileKeys)\n\n    // binned points\n    var binned = d3.histogram()(violinPointsValues)\n    // length of bins\n    var frequencies = binned.map(bin=>bin.length)\n    // min and max countour points for nice drawings\n    var minContourPoint = horizontalQ ? {x: 0, y: d3.min(violinPointsValues)} :  {x: d3.min(violinPointsValues), y: 0}\n    var maxContourPoint = horizontalQ ? {x: 0, y: d3.max(violinPointsValues)} :  {x: d3.max(violinPointsValues), y: 0}\n    var violinContourPoints = binned.map(function(bin, i) {\n        return horizontalQ\n        ? {y: (bin.length) ? d3.median(bin): d3.median([bin.x0, bin.x1]), x: frequencies[i]}\n        : {x: (bin.length) ? d3.median(bin): d3.median([bin.x0, bin.x1]), y: frequencies[i]}\n      })\n    // points along which to draw the violin shpe\n    violinContourPoints = [minContourPoint].concat(violinContourPoints).concat([maxContourPoint])\n\n    // set data\n    violinData.binned = binned;\n    violinData.frequencies = frequencies\n    violinData.contour = violinContourPoints\n    violinData.quartiles = pointQuartiles\n    violinData.pointKeys = violinPointsKeys\n    violinData.pointValues = violinPointsValues\n  }\n\n  return calculateViolinValues\n}\n","import {hypenate, safeSelect, round, interpolateColors} from './helpers';\nimport {setupContainer} from './utils';\nimport {colorFunction as CF} from './color-function';\n\n\nexport function numericLegend( selection ) {\n\n  var\n  min=0,\n  max=1,\n  spaceX,\n  spaceY,\n  colorFunction = CF(),\n  namespace='d3sm-linear-vertical-gradient',\n  fontSize = 12,\n  backgroundFill = 'transparent',\n  textColor = 'black',\n  roundTo = 2\n\n\n  legend.min = function(_) { return arguments.length ? (min=_, legend) : min }\n  legend.max = function(_) { return arguments.length ? (max=_, legend) : max }\n  legend.spaceX = function(_) { return arguments.length ? (spaceX=_, legend) : spaceX }\n  legend.spaceY = function(_) { return arguments.length ? (spaceY=_, legend) : spaceY }\n  legend.namespace = function(_) { return arguments.length ? (namespace=_, legend) : namespace }\n  legend.fontSize = function(_) { return arguments.length ? (fontSize=_, legend) : fontSize }\n  legend.backgroundFill = function(_) { return arguments.length ? (backgroundFill=_, legend) : backgroundFill }\n  legend.colorFunction = function(_) { return arguments.length ? (colorFunction=_, legend) : colorFunction }\n  legend.textColor = function(_) { return arguments.length ? (textColor=_, legend) : textColor }\n  legend.roundTo = function(_) { return arguments.length ? (roundTo=_, legend) : roundTo }\n\n  function legend() {\n    // background cliping rectangle\n    var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}\n    var container = setupContainer( selection, namespace, bgcpRect, backgroundFill );\n\n    var defs = safeSelect(selection, 'defs')\n    var linearGradient = safeSelect(defs, 'linearGradient')\n    .attr(\"x1\", \"0%\")\n    .attr(\"y1\", \"100%\")\n    .attr(\"x2\", \"0%\")\n    .attr(\"y2\", \"0%\")\n    .attr('id', hypenate(namespace,'numerical-legend-gradient'))\n\n\n    colorFunction.dataExtent([min, max])\n    .colorBy('value')\n    .valueExtractor(function(k, v, i){return v})\n\n\n    linearGradient.selectAll('stop')\n    .data( colorFunction.colors() )\n    .enter()\n    .append('stop')\n    .attr(\"offset\", function(d, i){ return i / (colorFunction.colors().length - 1) })\n    .attr('stop-color', function(d) {return d})\n\n\n\n\n    var rect = safeSelect(container, 'rect', 'legend')\n    .attr('transform', 'translate(0,'+fontSize+')')\n    .style(\"fill\", \"url(#\"+hypenate(namespace,'numerical-legend-gradient')+\")\")\n    .attr('x', 0)\n    .attr('y', 0)\n    .attr('width', spaceX)\n    .attr('height', spaceY - fontSize*2)\n    .on('mousemove', function(d, i){legendMousemove(d, i, rect, d3.select(this))})\n    .on('mouseout', function(d, i){ d3.select(\"#\"+hypenate(namespace,'legend-tooltip')).remove() })\n\n    var minText = safeSelect(container, 'text', 'min')\n    .text(round(min, 2))\n    .attr('text-anchor', 'middle')\n    .attr(\"font-size\", fontSize+'px')\n    .attr('transform', function(d, i){\n      var\n      x = spaceX / 2,\n      y = spaceY - fontSize / 4,\n      t = 'translate('+x+','+y+')'\n      return t\n    })\n\n    var maxText = safeSelect(container, 'text', 'max')\n    .text(round(max, 2))\n    .attr('text-anchor', 'middle')\n    .attr(\"font-size\", fontSize+'px')\n    .attr('transform', function(d, i){\n      var\n      x = spaceX / 2,\n      y = fontSize,\n      t = 'translate('+x+','+y+')'\n      return t\n    })\n\n\n\n\n  }\n\n  function legendMousemove(d, i, rect, t) {\n    var s = d3.scaleLinear()\n    .domain([0, rect.attr('height')])\n    .range([max, min])\n    var m = d3.mouse(rect.node())\n    var v = round(s(m[1]),roundTo)\n\n    var strokeColor = colorFunction(undefined, v, undefined, 'stroke')\n    var fillColor = colorFunction(undefined, v, undefined, 'fill')\n\n    var div = safeSelect(d3.select('body'), 'div', hypenate(namespace,'legend-tooltip'))\n    .attr('id', hypenate(namespace,'legend-tooltip'))\n    .style('position', 'absolute')\n    .style('left', (d3.event.pageX+15)+'px')\n    .style('top', (d3.event.pageY+15)+'px')\n    .style('background-color', fillColor)\n    .style('border-color', strokeColor)\n\n    .style('min-width', (fontSize * (String(max).split('.')[0].length+3))+'px')\n    .style('min-height', (fontSize * (String(max).split('.')[0].length+3))+'px')\n    .style('border-radius', '50%')\n    .style('border-radius', '5000px')\n\n    .style('display', 'flex')\n    .style('justify-content', 'center')\n    .style('text-align', 'middle')\n    .style('padding', 2+\"px\")\n\n    .style('border-style', 'solid')\n    .style('border-width', 2)\n\n    var text = safeSelect(div, 'div')\n    .text(v)\n    .style('color', textColor)\n    .style('align-self', 'center')\n  }\n\n  return legend\n}\n","import {hypenate, safeSelect, round, interpolateColors} from './helpers';\nimport {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer} from './utils';\nimport {colorFunction as CF} from './color-function';\nimport {groupingSpacer} from './grouping-spacer';\nimport {unique, flatten} from './array-functions';\n\nexport function categoricLegend( selection ) {\n  var\n  categories,\n\n  /**\n  * Data to plot. Assumed to be a object, where each key corresponds to a legend\n  * (see {@link legend#data})\n  * @param {Object} [data=undefined]\n  * @memberof legend#\n  * @property\n  */\n  data,\n  /**\n  * Which direction to render the bars in\n  * (see {@link legend#orient})\n  * @param {number} [orient='horizontal']\n  * @memberof legend#\n  * @property\n  */\n  orient='horizontal',\n  /**\n  * Amount of horizontal space (in pixels) avaible to render the legend in\n  * (see {@link legend#spaceX})\n  * @param {number} [spaceX=undefined]\n  * @memberof legend#\n  * @property\n  */\n  spaceX,\n  /**\n  * Amount of vertical space (in pixels) avaible to render the legend in\n  * (see {@link legend.spaceY})\n  * @param {number} [spaceY=undefined]\n  * @memberof legend#\n  * @property\n  */\n  spaceY,\n\n  /**\n  * Whether or not to allow legend to render elements pass the main spatial dimension\n  * given the orientation (see {@link legend#orient}), where {@link legend#orient}=\"horizontal\"\n  * the main dimension is {@link legend#spaceX} and where {@link legend#orient}=\"vertical\"\n  * the main dimension is {@link legend#spaceY}\n  * @param {boolean} [overflowQ=false]\n  * @memberof legend#\n  * @property\n  */\n  overflowQ = false,\n\n  /**\n  * An array - putatively of other arrays - depicting how bars should be arranged\n  * @param {Array[]} [grouping=undefined]\n  * @memberof legend#\n  * @property\n  */\n  grouping,\n\n  /**\n  * How to get the value of the legend\n  * @param {function} [valueExtractor=function(key, index) { return data[key] }]\n  * @memberof legend#\n  * @property\n  */\n  valueExtractor = function(key, index) { return data[key] },\n  /**\n  * How to sort the bars - if {@link bar#grouping} is not provided.\n  * @param {function} [sortingFunction=function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}]\n  * @memberof bar#\n  * @property\n  */\n  sortingFunction = function(keyA, keyB) {return d3.ascending(keyA, keyB)},\n  /**\n  * Default space for the spacer (percentage) of main dimension given the orientation\n  * (see {@link legend#orient}), where {@link legend#orient}=\"horizontal\"\n  * the main dimension is {@link legend#spaceX} and where {@link legend#orient}=\"vertical\"\n  * the main dimension is {@link legend#spaceY} between bars\n  * @param {number} [objectSpacer=0.05]\n  * @memberof legend#\n  * @property\n  */\n  objectSpacer = 0.05,\n  /**\n  * The minimum size that an object can be\n  * @param {number} [minObjectSize=50]\n  * @memberof legend#\n  * @property\n  */\n  minObjectSize = 10,\n  /**\n  * The maximum size that an object can be\n  * @param {number} [maxObjectSize=100]\n  * @memberof legend#\n  * @property\n  */\n  maxObjectSize = 100,\n\n  /**\n  * The stroke width of the bars\n  * @param {number} [barStrokeWidth=2]\n  * @memberof legend#\n  * @property\n  */\n  bubbleStrokeWidth = 2,\n  /**\n  * Instance of ColorFunction\n  * @param {function} [colorFunction = colorFunction()]\n  * @memberof legend#\n  * @property\n  */\n  colorFunction = CF(),\n\n  /**\n  * Color of the background\n  * @param {string} [backgroundFill=\"transparent\"]\n  * @memberof legend#\n  * @property\n  */\n  backgroundFill = 'transparent',\n  /**\n  * Namespace for all items made by this instance of legend\n  * @param {string} [namespace=\"d3sm-legend\"]\n  * @memberof legend#\n  * @property\n  */\n  namespace = 'd3sm-legend',\n  /**\n  * Class name for legend container (<g> element)\n  * @param {string} [objectClass=\"legend\"]\n  * @memberof legend#\n  * @property\n  */\n  objectClass = 'legend',\n\n  /**\n  * Duration of all transitions of this element\n  * @param {number} [transitionDuration=1000]\n  * @memberof legend#\n  * @property\n  */\n  transitionDuration = 1000,\n  /**\n  * Easing function for transitions\n  * @param {d3.ease} [easeFunc=d3.easeExp]\n  * @memberof legend#\n  * @property\n  */\n  easeFunc = d3.easeExp\n\n\n  legend.categories = function(_) { return arguments.length ? (categories=_, legend) : categories };\n\n  /**\n   * Gets or sets the selection in which items are manipulated\n   * @param {d3.selection} [_=none]\n   * @returns {legend | d3.selection}\n   * @memberof legend\n   * @property\n   * by default selection = selection\n   */\n  legend.selection = function(_) { return arguments.length ? (selection = _, legend) : selection; };\n  /**\n   * Gets or sets the data\n   * (see {@link legend#data})\n   * @param {number} [_=none]\n   * @returns {legend | object}\n   * @memberof legend\n   * @property\n   */\n  legend.data = function(_) { return arguments.length ? (data = _, legend) : data; };\n  /**\n   * Gets or sets the orient of the bars\n   * (see {@link legend#orient})\n   * @param {number} [_=none]\n   * @returns {legend | object}\n   * @memberof legend\n   * @property\n   */\n  legend.orient = function(_) { return arguments.length ? (orient = _, legend) : orient; };\n  /**\n   * Gets or sets the amount of horizontal space in which items are manipulated\n   * (see {@link legend#spaceX})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {legend | number}\n   * @memberof legend\n   * @property\n   * by default spaceX = undefined\n   */\n  legend.spaceX = function(_) { return arguments.length ? (spaceX = _, legend) : spaceX; };\n  /**\n   * Gets or sets the amount of vertical space in which items are manipulated\n   * (see {@link legend#spaceY})\n   * @param {number} [_=none] should be a number > 0\n   * @returns {legend | number}\n   * @memberof legend\n   * @property\n   * by default spaceY = undefined\n   */\n  legend.spaceY = function(_) { return arguments.length ? (spaceY = _, legend) : spaceY; };\n\n  /**\n   * Gets / sets whether or not legend is allowed to go beyond specified dimensions\n   * (see {@link legend#spaceX})\n   * @param {boolean} [_=none]\n   * @returns {legend | boolean}\n   * @memberof legend\n   * @property\n   * by default overflowQ = false\n   */\n  legend.overflowQ = function(_) { return arguments.length ? (overflowQ = _, legend) : overflowQ; };\n  /**\n   * Gets / sets the grouping of the bars\n   * (see {@link legend#grouping})\n   * @param {Array[]} [_=none]\n   * @returns {legend | Array[]}\n   * @memberof legend\n   * @property\n   * by default grouping = undefined\n   */\n  legend.grouping = function(_) { return arguments.length ? (grouping = _, legend) : grouping; };\n  /**\n   * Gets / sets the valueExtractor\n   * (see {@link legend#valueExtractor})\n   * @param {function} [_=none]\n   * @returns {legend | function}\n   * @memberof legend\n   * @property\n   * by default valueExtractor = function(key, index) { return data[key] },\n   */\n  legend.valueExtractor = function(_) { return arguments.length ? (valueExtractor = _, legend) : valueExtractor; };\n  /**\n   * Gets / sets the sortingFunction\n   * (see {@link bar#sortingFunction})\n   * @param {function} [_=none]\n   * @returns {bar | function}\n   * @memberof bar\n   * @property\n   * by default sortingFunction = function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])},\n   */\n  legend.sortingFunction = function(_) { return arguments.length ? (sortingFunction = _, legend) : sortingFunction; };\n  /**\n   * Gets / sets objectSpacer\n   * (see {@link legend#objectSpacer})\n   * @param {number} [_=none]\n   * @returns {legend | number}\n   * @memberof legend\n   * @property\n   * by default objectSpacer = 0.05\n   */\n  legend.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, legend) : objectSpacer; };\n  /**\n   * Gets / sets the minObjectSize\n   * (see {@link legend#minObjectSize})\n   * @param {number} [_=none]\n   * @returns {legend | number}\n   * @memberof legend\n   * @property\n   * by default minObjectSize = 50\n   */\n  legend.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, legend) : minObjectSize; };\n  /**\n   * Gets / sets the maxObjectSize\n   * (see {@link legend#maxObjectSize})\n   * @param {number} [_=none]\n   * @returns {legend | number}\n   * @memberof legend\n   * @property\n   * by default maxObjectSize = 100\n   */\n  legend.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, legend) : maxObjectSize; };\n\n  /**\n   * Gets / sets the barStrokeWidth\n   * (see {@link legend#barStrokeWidth})\n   * @param {number} [_=none]\n   * @returns {legend | number}\n   * @memberof legend\n   * @property\n   * by default barStrokeWidth = 2\n   */\n  legend.bubbleStrokeWidth = function(_) { return arguments.length ? (bubbleStrokeWidth = _, legend) : bubbleStrokeWidth; };\n  /**\n   * Gets / sets the colorFunction\n   * (see {@link legend#colorFunction})\n   * @param {number} [_=none]\n   * @returns {legend | number}\n   * @memberof legend\n   * @property\n   * by default colorFunction = colorFunction()\n   */\n  legend.colorFunction = function(_) { return arguments.length ? (colorFunction = _, legend) : colorFunction; };\n\n  /**\n   * Gets / sets the backgroundFill\n   * (see {@link legend#backgroundFill})\n   * @param {string} [_=none]\n   * @returns {legend | string}\n   * @memberof legend\n   * @property\n   * by default backgroundFill = 'transparent'\n   */\n  legend.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, legend) : backgroundFill; };\n  /**\n   * Gets / sets the namespace\n   * (see {@link legend#namespace})\n   * @param {string} [_=none]\n   * @returns {legend | string}\n   * @memberof legend\n   * @property\n   * by default namespace = 'd3sm-legend'\n   */\n  legend.namespace = function(_) { return arguments.length ? (namespace = _, legend) : namespace; };\n  /**\n   * Gets / sets the objectClass\n   * (see {@link legend#objectClass})\n   * @param {string} [_=none]\n   * @returns {legend | string}\n   * @memberof legend\n   * @property\n   * by default objectClass = 'tick-group'\n   */\n  legend.objectClass = function(_) { return arguments.length ? (objectClass = _, legend) : objectClass; };\n  /**\n   * Gets / sets the transitionDuration\n   * (see {@link legend#transitionDuration})\n   * @param {number} [_=none]\n   * @returns {legend | number}\n   * @memberof legend\n   * @property\n   * by default transitionDuration = 1000\n   */\n  legend.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, legend) : transitionDuration; };\n  /**\n   * Gets / sets the easeFunc\n   * (see {@link legend#easeFunc})\n   * @param {d3.ease} [_=none]\n   * @returns {legend | d3.ease}\n   * @memberof legend\n   * @property\n   * by default easeFunc = d3.easeExp\n   */\n  legend.easeFunc = function(_) { return arguments.length ? (easeFunc = _, legend) : easeFunc; };\n\n\n  function legend() {\n    var horizontalQ = (orient == 'horizontal') ? true : false\n    var verticalQ = !horizontalQ\n    // background cliping rectangle\n    var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}\n    var container = setupContainer( selection, namespace, bgcpRect, backgroundFill );\n\n\n    colorFunction.dataExtent([0, categories.length - 1])\n    .colorBy('categories')\n    .categoryExtractor(function(k, v, i){return v})\n\n    var r = Math.min(spaceX, spaceY) / 2\n    var numberOfObjects = categories.length\n\n    // if grouping is undefined sort barKeys by sortingFunction\n    var ordered = (grouping == undefined) ? categories.sort(sortingFunction) : grouping\n    // ordered might be nested depending on grouping\n    var catKeys = flatten(ordered)\n\n    var space = horizontalQ ? spaceX : spaceY\n    // calculate object size\n    var objectSize = calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ)\n    // calculate spacer size if needed\n    var spacerSize = calculateWidthOfSpacer(catKeys, space, objectSize, numberOfObjects, objectSpacer, overflowQ)\n    // make the nested groups\n    var spacerFunction = groupingSpacer()\n    .horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects)\n    .objectClass(objectClass).objectSize(objectSize).spacerSize(spacerSize)\n    .transitionDuration(transitionDuration).easeFunc(easeFunc)\n    .namespace(namespace)\n\n    spacerFunction(container, ordered, 0)\n    var r = Math.min(objectSize, spaceX, spaceY) / 2 - bubbleStrokeWidth\n\n    container.selectAll('g:not(.to-remove).'+objectClass).each(function(cat, i) {\n      var t = d3.select(this)\n      var c = safeSelect(t, 'circle')\n      var fillColor = colorFunction(undefined, cat, i, 'fill'), // prevent duplicate computation\n      strokeColor = colorFunction(undefined, cat, i,  'stroke')\n\n      var cx = horizontalQ\n        ? r+bubbleStrokeWidth\n        : (spaceX - r*2) / 2 + r\n      var cy = verticalQ\n        ? r+bubbleStrokeWidth\n        : (spaceX - r*2) / 2 + r\n\n      c.attr(\"r\", r)\n      .attr('cx', cx)\n      .attr('cy', cy)\n      .attr('fill', fillColor)\n      .attr('stroke', strokeColor)\n      .attr('stroke-width', bubbleStrokeWidth)\n\n      var text = safeSelect(t, 'text')\n      text.text(cat)\n      .attr('text-anchor', 'middle')\n      .attr(\"transform\", function(d, i){\n        var\n        x = cx,\n        y = cy + text.node().getBoundingClientRect().height / 4,\n        t = 'translate('+x+','+y+')'\n        return t\n      })\n\n    })\n\n\n  }\n\n  return legend\n}\n","import {hypenate, safeSelect, modifyHexidecimalColorLuminance, extractViolinValues, quartiles} from './helpers';\nimport {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer} from './utils';\nimport {unique, hasQ, flatten, whichBin} from './array-functions';\nimport {groupingSpacer} from './grouping-spacer';\nimport {colorFunction as CF} from './color-function';\nimport {tooltip as TTip} from './tooltip';\nimport {lasso} from './lasso';\nimport './d3-prototypes';\n\nexport function lassoWidget( selection ) {\n  var\n  namespace = 'd3sm-lasso',\n  selection = selection,\n  svg,\n  chartContainer,\n  objectContainer,\n  objectClass,\n  lassoLine = d3.line()\n  .x(function(d, i){return d[0]})\n  .y(function(d, i){return d[1]})\n  .curve(d3.curveLinearClosed),\n  path,\n  colorFunction = CF(),\n  maxNumberOfGroups,\n  dataExtractor,\n  xScale,\n  yScale\n\n  function onError(msg, style){\n    console.log(msg, style)\n  }\n  // setup the container\n  setupDataselectContainer()\n\n  lassoWidget.objectClass = function(_){return arguments.length ? (objectClass='.'+_.replace('.',''), lassoWidget) : objectClass}\n  lassoWidget.svg = function(_){return arguments.length ? (svg=_, lassoWidget) : svg}\n  lassoWidget.submit = function(_){return arguments.length ? (submit=_, lassoWidget) : submit}\n  lassoWidget.maxNumberOfGroups = function(_){return arguments.length ? (maxNumberOfGroups=_, lassoWidget) : maxNumberOfGroups}\n  lassoWidget.onError = function(_){return arguments.length ? (onError=_, lassoWidget) : onError}\n  lassoWidget.objectContainer = function(_){return arguments.length ? (objectContainer=_, lassoWidget) : objectContainer}\n  lassoWidget.chartContainer = function(_){return arguments.length ? (chartContainer=_, lassoWidget) : chartContainer}\n  lassoWidget.dataExtractor = function(_){return arguments.length ? (dataExtractor=_, lassoWidget) : dataExtractor}\n  lassoWidget.xScale = function(_){return arguments.length ? (xScale=_, lassoWidget) : xScale}\n  lassoWidget.yScale = function(_){return arguments.length ? (yScale=_, lassoWidget) : yScale}\n\n  function styles() {\n    var s = d3.select(\"html\").select(\"style.\"+namespace+'lasso-widget')\n    if (s.empty()) {\n      d3.select(\"html\").append(\"style\")\n      .classed(namespace+'lasso-widget', true)\n      .html(\n        \".\"+hypenate(namespace, \"data-table\") + \"{\\\n          height:100px;\\\n          overflow:auto;\\\n        }\"\n      )\n    }\n  }\n\n\n  function lassoWidget() {\n    styles()\n\n  }\n\n  function submit(data) {\n    console.log(data)\n  }\n\n  function plusTab(tabList) {\n    var tab = safeSelect(tabList, 'li', hypenate(namespace, 'plus-tab'))\n    .classed('ml-auto', true)\n    .classed('nav-item', true)\n    .on('click', makeNewGroup),\n\n    anchor = safeSelect(tab, 'a', 'nav-link'),\n    icon = safeSelect(anchor, 'i', 'fa')\n    .classed('fa-plus fa-2x text-success', true)\n  }\n\n  function sendTab(tabList) {\n    var tab = safeSelect(tabList, 'li', hypenate(namespace, 'send-tab'))\n    .classed('ml-auto', true)\n    .classed('nav-item', true)\n    .on('click', clickSend)\n    ,\n\n    anchor = safeSelect(tab, 'a', 'nav-link'),\n    icon = safeSelect(anchor, 'i', 'fa')\n    .classed('fa-paper-plane-o fa-2x text-primary', true)\n  }\n\n  function clickSend() {\n    var data = gatherDataLists()\n    submit(data)\n  }\n\n  function closeTab(tabList) {\n    var tab = safeSelect(tabList, 'li', hypenate(namespace, 'close-tab'))\n    .classed('ml-auto', true)\n    .classed('nav-item', true)\n    .on('click', function(){\n      var m = d3.mouse(d3.select(\"html\").node())\n      selection.select('div.card').style(\"position\", 'relative')\n      .transition().duration(1000)\n      .ease(d3.easeBack)\n      .style('left', window.outerWidth +'px')\n      .remove()\n    })\n    ,\n\n    anchor = safeSelect(tab, 'a', 'nav-link'),\n    icon = safeSelect(anchor, 'i', 'fa')\n    .classed('fa-window-close-o fa-2x text-danger', true)\n  }\n\n\n\n  function setupDataselectContainer() {\n    var\n    card = safeSelect(selection, 'div', 'card'),\n    cardHeader = safeSelect(card, 'div', 'card-header'),\n\n    tabList = safeSelect(cardHeader, 'ul', 'nav')\n    .classed('nav-tabs card-header-tabs', true)\n    .classed(hypenate(namespace, 'tab-list'), true)\n    .attr('role', 'tablist'),\n\n    cardBody = safeSelect(card, 'div', 'card-body'),\n    tabContent = safeSelect(cardBody, 'div', 'tab-content')\n    .classed(hypenate(namespace, 'tab-content'), true),\n\n    // leftTabs = safeSelect(tabList, 'div', 'nav mr-auto left-aligned-tabs')\n    rightTabs = safeSelect(tabList, 'div', 'right-aligned-tabs').classed('nav ml-auto ', true)\n\n    plusTab(rightTabs)\n    sendTab(rightTabs)\n    closeTab(rightTabs)\n    var\n    defaultTab = safeSelect(tabContent, 'div', 'tab-pane')\n    .classed(hypenate(namespace,'default-tab'), true)\n    .classed(\"active\", true)\n    .classed('text-left', true),\n\n    p = safeSelect(defaultTab, 'div')\n    .html(\n      \"Click <kbd><i class='fa fa-plus text-success'></i></kbd> to add a new group.<br>\"+\n      \"Click <kbd><i class='fa fa-paper-plane-o text-primary'></i></kbd> to submit for re-analysis.<br>\"+\n      \"Click <kbd><i class='fa fa-window-close-o text-danger'></i></kbd> to close the dataselect widget.\"\n    )\n    // .text('Click the green plus to add a new group.')\n  }\n\n\n  function makeRemoveButton(paneBtnList) {\n    var\n    btn = safeSelect(paneBtnList, 'button', 'remove-btn')\n    .classed('btn btn-danger', true)\n    .on('click', removeBtnClickToRemove),\n    i = safeSelect(btn, 'i', 'fa').classed('fa-trash-o', true),\n    s = safeSelect(btn, 'span').text('Remove group')\n    return btn\n  }\n\n  function removeBtnClickToRemove(d, i) {\n    var\n    tab = selection.select('#'+hypenate(namespace,'tab',d)),\n    pane = selection.select('#'+hypenate(namespace,'tab','pane',d))\n\n    tab.remove()\n    pane.remove()\n\n    showRemainingPanes()\n    updateTabAndPaneNumbers()\n  }\n\n\n  function togglePaneById(id) {\n    var panes = selection.select('.'+hypenate(namespace, 'tab-content'))\n    panes.selectAll('div.tab-pane')\n    .each(function(d, i){\n      d3.select(this).classed('active show', d3.select(this).attr('id') == id)\n    })\n  }\n\n  function insertTab(tabs, n) {\n\n    // var li = tabs.insert(\"li\", ':nth-child('+(n)+')')\n    var li = tabs.insert(\"li\", '.right-aligned-tabs')\n    .classed('nav-item', true)\n    .classed('alert', true)\n    .classed('alert-secondary', true)\n    .classed('alert-dismissable', true)\n    .classed('fade', true)\n    .classed('show', true)\n    // .classed('mr-auto', true)\n    .attr(\"role\", 'alert')\n    .attr(\"id\", hypenate(namespace,'tab',n))\n    .classed(hypenate(namespace,'tab'), true)\n\n    var a = safeSelect(li, 'a')\n    .attr('data-toggle', 'tab')\n    .text('Group ' + n)\n    .attr('href', '#'+hypenate(namespace,'tab','pane',n))\n    .on('dblclick', function(d, i){\n      var t = d3.select(this)\n      t.attr('contenteditable', true)\n      d3.select(t.node().parentNode)\n      .classed('alert-secondary', false)\n      .classed('alert-warning', true)\n\n      // d3.select(\"html\").on(\"click\", dispatchBlur)\n      //\n      // function dispatchBlur(d, i) {\n      //   if (d3.event.target != d3.select(this)) {\n      //     d3.select(this).dispatch(\"blur\")\n      //   }\n      // }\n\n\n    })\n    .on('blur', function(d, i){\n\n      var t = d3.select(this)\n      t.attr('contenteditable', false)\n      d3.select(t.node().parentNode)\n      .classed('alert-secondary', true)\n      .classed('alert-warning', false)\n\n    })\n    .on('input', function(d, i){\n      var t = d3.select(this)\n      var txt = t.text()\n      d3.select(t.attr('href')).select(\"p.lead\").text(txt)\n    })\n\n\n    return li\n  }\n\n  function makeTabPane(panes, n) {\n    var pane = panes.append('div')\n    .datum(n)\n    .attr('role', 'tabpanel')\n    .classed('tab-pane', true)\n    .classed('text-left', true)\n    .classed(hypenate(namespace,'tab','pane'), true)\n    .attr('id', hypenate(namespace,'tab','pane',n))\n    return pane\n  }\n\n  function populatePane(pane, n) {\n    var\n    lead = safeSelect(pane, 'p', 'lead').text('Group '+n),\n\n    tableContainer = safeSelect(pane, 'div', 'table-responsive')\n    .attr(\"class\", hypenate(namespace, \"data-table\")),\n\n    table = safeSelect(tableContainer, \"table\", \"table\")\n    .classed(\"table-sm\", true).classed(\"table-hover\", true),\n\n    caption = safeSelect(table, \"caption\", \"caption\").html(\"List of selected\"),\n\n    btns = safeSelect(pane, 'div', 'text-right'),\n\n    cN = n % colorFunction.colors().length\n    if (n % 2 != 0) { cN = (colorFunction.colors().length - 1) - cN }\n\n\n\n\n\n    var lassoBtn = makeLassoButton(btns)\n    var currentLasso = makeLassoFunction(n, colorFunction.colors()[cN])\n    var clearBtn = makeClearButton(btns)\n\n    bindButtons(lassoBtn, clearBtn, currentLasso, table)\n\n    var removeBtn = makeRemoveButton(btns)\n\n    lead.on(\"mouseover\", function(){\n      currentLasso.draw()\n    })\n    lead.on(\"mouseout\", function(){\n      currentLasso.remove()\n    })\n\n  }\n\n  function makeLassoButton(btns) {\n    var btn = safeSelect(btns, 'button', 'lasso-btn')\n    .classed('btn btn-info', true)\n    .classed(namespace, true)\n    // .datum([lasso])\n    var i = safeSelect(btn, 'i', 'fa').classed('fa-hand-pointer-o', true)\n    var s = safeSelect(btn, 'span').text('Lasso select')\n    return btn\n\n  }\n\n  function makeClearButton(btns) {\n    var btn = safeSelect(btns, 'button', 'clear-btn')\n    .classed('btn btn-light', true)\n    .classed(namespace, true)\n    var i = safeSelect(btn, 'i', 'fa').classed('fa-eraser', true)\n    var s = safeSelect(btn, 'span').text('Clear selection')\n    return btn\n  }\n\n  function makeLassoFunction(instance, color) {\n    var currentLasso = lasso()\n    .namespace(namespace)\n    .svg(svg)\n    .objectClass(objectClass)\n    .chartContainer(chartContainer)\n    .objectContainer(objectContainer)\n    .instance(instance)\n    .color(color)\n    .yScale(yScale)\n    .xScale(xScale)\n\n    return currentLasso\n  }\n\n  function bindButtons(lassoBtn, clearBtn, currentLasso, table) {\n    currentLasso.eventCatcher(lassoBtn)\n    lassoBtn.node().addEventListener(\"click\", lassoBtnToggle)\n\n    lassoBtn.datum([currentLasso])\n    .on(hypenate(namespace,'render'), function(){\n        d3.select(this).datum()[0].draw()\n    })\n    .on(hypenate(namespace,\"drag\"), function(las){\n      var nodes = chartContainer.selectAll(\".in-lasso-\"+las[0].instance()).nodes()\n      var tableData = nodes.map(function(d, i){\n        var extracted = dataExtractor(d3.select(d).datum())\n        extracted[\"__node\"] = d\n        return extracted\n      })\n\n\n      makeTable(table, tableData, currentLasso)\n    })\n\n    clearBtn.on('click', function(){\n      currentLasso.allPoints([])\n      currentLasso.currentPoints([])\n      lassoBtn.dispatch(hypenate(namespace,\"drag\"))\n\n    })\n\n  }\n\n  function lassoBtnToggle() {\n    var that = d3.select(this)\n    var las = that.datum()[0]\n\n    las.toggle()\n    var activeQ = las.activeQ()\n\n    if (las.activeQ()) {\n      that.classed('btn-info', !activeQ)\n      that.classed('btn-warning', activeQ)\n      that.select(\"span\").text(\"Lasso select (active)\")\n      selection.selectAll(\".\"+namespace+\".lasso-btn\").dispatch(hypenate(namespace,'render'))\n      d3.select(\"html\").node().addEventListener('mousedown', monitorLassoButtonState)\n    } else {\n      that.classed('btn-info', !activeQ)\n      that.classed('btn-warning', activeQ)\n      that.select(\"span\").text(\"Lasso select\")\n      d3.select(\"html\").node().removeEventListener('mousedown', monitorLassoButtonState)\n    }\n\n    function monitorLassoButtonState(event) {\n      /*\n      activeLasso stops event stopPropagation, so this event will not register in\n      that case. Thus we only need to ensure that the usre is clicking on the lasso button\n      otherwise, de-spawn lasso.\n      */\n      if (\n        event.target != that.node() &&\n        event.target != that.select(\"span\").node() &&\n        event.target != that.select(\"i\").node()\n      ) {\n        // event.preventDefault()\n        // event.stopPropagation()\n        las.toggle(false)\n        that.classed('btn-info', true)\n        that.classed('btn-warning', false)\n        that.select(\"span\").text(\"Lasso select\")\n        // console.log(that, that.node())\n        // that.dispatch(\"focusout\")\n      }\n    }\n\n  }\n\n\n  function updateTableHeaderColumns(headR, tableData) {\n    var headerKeys = d3.keys(tableData[0]).filter(k=>k!=\"__node\")\n    if (headerKeys.length > 0) {\n      // headerKeys = [\"remove\"].concat(headerKeys)\n      headerKeys.push(\"remove\")\n    }\n\n    var headerCols = headR.selectAll(\"th\")\n\n    headerCols = headerCols.data(headerKeys)\n    headerCols.exit().remove()\n    headerCols = headerCols.merge(headerCols.enter().append(\"th\").attr(\"scope\",\"col\"))\n    .text(function(d, i){return d})\n\n    return headerKeys\n  }\n\n  function updateTableRows(body, tableData) {\n    var bodyRows = body.selectAll(\"tr\")\n\n    bodyRows = bodyRows.data(tableData)\n    bodyRows.exit().remove()\n    bodyRows = bodyRows.merge(bodyRows.enter().append(\"tr\"))\n\n    return bodyRows\n  }\n\n  function updateTableRowColumns(cols, headerKeys, rowData, tableData, lasso, table) {\n    cols = cols.data(headerKeys)\n    cols.exit().remove()\n    cols = cols.merge(cols.enter().append(\"td\"))\n\n    cols.html(function(d, i){\n      if (d != \"remove\") { return rowData[d] }\n      return \"<i class='fa fa-close'></i>\"\n    }).classed(\"text-left\", function(d, i){\n      if (d != \"remove\") { return false }\n      return true\n    })\n    // .attr(\"scope\",function(d, i){\n    //   if (d != \"remove\") { return false }\n    //   return true\n    // })\n\n    cols.select(\"i.fa-close\").on(\"click\", function(d, i){\n      var node = rowData[\"__node\"],\n      n = d3.select(node)\n      n.classed(\"in-lasso\", false)\n      n.classed(\"in-lasso-\"+lasso.instance(), false)\n\n      tableData.map(function(dd, j){\n        if (dd[\"__node\"] == node) {\n          tableData.splice(j, 1)\n          makeTable(table, tableData, lasso)\n        }\n      })\n\n    })\n  }\n\n  function makeTable(table, tableData, lasso) {\n    var\n    head = safeSelect(table, \"thead\"),\n    headR = safeSelect(head, \"tr\"),\n    body = safeSelect(table, \"tbody\"),\n\n    headerKeys = updateTableHeaderColumns(headR, tableData),\n\n    bodyRows = updateTableRows(body, tableData)\n\n    bodyRows.each(function(rowData, i){\n      var t = d3.select(this)\n      var cols = t.selectAll(\"td\")\n\n      updateTableRowColumns(cols, headerKeys, rowData, tableData, lasso, table)\n\n      t.on(\"mouseover\", function(d, i) {\n        lasso.applyObjectAttributes(d3.select(d[\"__node\"]),true)\n      })\n      .on(\"mouseout\", function(d, i) {\n        lasso.applyObjectAttributes(d3.select(d[\"__node\"]),false)\n      })\n    })\n\n\n  }\n\n\n\n\n\n\n\n  function makeNewGroup(d, i) {\n\n    var tabs = selection.select('.'+hypenate(namespace, 'tab-list')),\n    panes = selection.select('.'+hypenate(namespace, 'tab-content')),\n    n = newGroupNumber()\n\n    if (tabs.selectAll('.'+hypenate(namespace,'tab')).size() == maxNumberOfGroups) {\n      onError('only '+maxNumberOfGroups+' allowed.', 'warning')\n      return\n    }\n\n    var\n    tab = insertTab(tabs, n),\n    pane = makeTabPane(panes, n)\n\n    populatePane(pane, n)\n    togglePaneById(pane.attr('id'))\n\n  }\n\n  function newGroupNumber() {\n    var tabs = selection.select('.'+hypenate(namespace, 'tab-list'))//,\n    var\n    n = tabs.selectAll('li').size() - 3, // minus 1 for plus tab\n    s = 'Group ' + n,\n    gs = []\n    tabs.each(function(d, i){ return gs.push(d3.select(this).select(\"a\").text()) })\n    while (gs.includes(s)) { n+=1; s = 'Group ' + n; }\n    return n\n  }\n\n  function updateTabAndPaneNumbers() {\n    var\n    tabs = selection.select('.'+hypenate(namespace, 'tab-list')),\n    panes = selection.select('.'+hypenate(namespace, 'tab-content'))\n\n    tabs = tabs.selectAll('.'+hypenate(namespace,'tab'))\n    panes = panes.selectAll('.'+hypenate(namespace,'tab', 'pane'))\n\n    tabs.each(function(d, i){\n      d3.select(this).datum(i)\n      .attr(\"id\", hypenate(namespace,'tab',i))\n      .select('a')\n      .attr('href', '#'+hypenate(namespace,'tab','pane',i))\n      .text(function(dd, ii){\n        var curText = d3.select(this).text();\n        if (curText.split(' ')[0] == 'Group') {\n            return 'Group ' + i\n        }\n        return curText\n      })\n    })\n\n    panes.each(function(d, i){\n      d3.select(this).datum(i)\n      .attr('id', hypenate(namespace,'tab','pane',i))\n      safeSelect(d3.select(this), 'p', 'lead')\n      .text(function(dd, ii){\n        var curText = d3.select(this).text();\n        if (curText.split(' ')[0] == 'Group') {\n            return 'Group ' + i\n        }\n        return curText\n      })\n      safeSelect(d3.select(this), 'button', 'remove-btn')\n      .on('click', removeBtnClickToRemove)\n    })\n\n  }\n\n  function gatherDataLists() {\n    var tabs = selection.select('.'+hypenate(namespace, 'tab-list'))\n    var panes = selection.select('.'+hypenate(namespace, 'tab-content'))\n    var tables = panes.selectAll('.'+hypenate(namespace, \"data-table\"))\n    var data = {}\n\n    var textGroups = tabs.selectAll('li.'+hypenate(namespace,'tab') + ' > a')\n    .nodes().map(function(d, i){return d3.select(d).text()})\n\n    textGroups.map(function(e, i){\n      data[e] = []\n    })\n\n\n    tables.each(function(d, i){\n      var table = d3.select(this).select(\"tbody\")\n      data[textGroups[i]] = table.selectAll('tr').data()\n    })\n\n    return data\n  }\n\n  function showRemainingPanes() {\n    var\n    tabs = selection.select('.'+hypenate(namespace, 'tab-list')),\n    panes = selection.select('.'+hypenate(namespace, 'tab-content')),\n    remainingTabs = tabs.selectAll('.'+hypenate(namespace,'tab'))\n\n    if (remainingTabs.size() == 0) {\n      panes.select('.'+hypenate(namespace,'default-tab'))\n      .classed(\"active\", true)\n      .classed('text-left', true)\n    }\n    else {\n      var lastTab = remainingTabs.nodes()[remainingTabs.size()-1],\n      lastPaneId = d3.select(lastTab).select('a').attr('href')\n      panes.select(lastPaneId)\n      .classed(\"active\", true)\n      .classed('text-left', true)\n    }\n  }\n\n\n  return lassoWidget\n}\n","import {hypenate, safeSelect} from './helpers';\nimport {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer, log} from './utils';\nimport {unique, flatten} from './array-functions';\nimport {groupingSpacer} from './grouping-spacer';\nimport {colorFunction as CF} from './color-function';\nimport {tooltip as TTip} from './tooltip';\nexport function upset ( selection ) {\n  var\n  data,\n  orient='horizontal',\n  spaceX,\n  spaceY,\n  overflowQ=false,\n  minObjectSize=20,\n  maxObjectSize=50,\n  circleStrokeWidth=2,\n  // colorFunction=\n  backgroundFill = 'transparent',\n  namespace='d3sm-upset',\n  objectClass = 'upset',\n\n  transitionDuration = 1000,\n  easeFunc = d3.easeExp,\n\n  setKey = \"set\",\n  intersectionKey = \"intersection\",\n  elementsKey = \"elements\",\n\n  setExtractor = function(key, i) {return data[key][setKey]},\n  intersectionExtractor = function(key, i) {return data[key][intersectionKey]},\n  elementExtractor = function(key, i) {return data[key][elementsKey]},\n\n  cellKeys,\n  setValues,\n  intersectionValues,\n  elementValues,\n\n  xObjectSpacer = 0.05,\n  yObjectSpacer = 0.05,\n  radius,\n\n  // listDelim = ';'\n\n  yObjectSize,\n  ySpacerSize,\n  xObjectSize,\n  xSpacerSize,\n\n  setKeySortingFunction = function(a, b) { return setValues.indexOf(setExtractor(a)) - setValues.indexOf(setExtractor(b)) },\n  intersectionKeySortingFunction = function(a, b) { return intersectionValues.indexOf(intersectionExtractor(a)) - intersectionValues.indexOf(intersectionExtractor(b)) }\n\n\n  upset.selection = function(_) { return arguments.length ? (selection = _, upset) : selection; };\n  upset.data = function(_) { return arguments.length ? (data = _, upset) : data; };\n  upset.orient = function(_) { return arguments.length ? (orient = _, upset) : orient; };\n  upset.spaceX = function(_) { return arguments.length ? (spaceX = _, upset) : spaceX; };\n  upset.spaceY = function(_) { return arguments.length ? (spaceY = _, upset) : spaceY; };\n  upset.overflowQ = function(_) { return arguments.length ? (overflowQ = _, upset) : overflowQ; };\n  upset.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, upset) : minObjectSize; };\n  upset.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, upset) : maxObjectSize; };\n  upset.circleStrokeWidth = function(_) { return arguments.length ? (circleStrokeWidth = _, upset) : circleStrokeWidth; };\n  upset.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, upset) : backgroundFill; };\n  upset.namespace = function(_) { return arguments.length ? (namespace = _, upset) : namespace; };\n  upset.objectClass = function(_) { return arguments.length ? (objectClass = _, upset) : objectClass; };\n  upset.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, upset) : transitionDuration; };\n  upset.easeFunc = function(_) { return arguments.length ? (easeFunc = _, upset) : easeFunc; };\n  upset.cellKeys = function(_) { return arguments.length ? (cellKeys = _, upset) : cellKeys; };\n  upset.setValues = function(_) { return arguments.length ? (setValues = _, upset) : setValues; };\n  upset.intersectionValues = function(_) { return arguments.length ? (intersectionValues = _, upset) : intersectionValues; };\n  upset.xObjectSpacer = function(_) { return arguments.length ? (xObjectSpacer = _, upset) : xObjectSpacer; };\n  upset.yObjectSpacer = function(_) { return arguments.length ? (yObjectSpacer = _, upset) : yObjectSpacer; };\n  upset.radius = function(_) { return arguments.length ? (radius = _, upset) : radius; };\n  upset.setExtractor = function(_) { return arguments.length ? (setExtractor = _, upset) : setExtractor; };\n  upset.intersectionExtractor = function(_) { return arguments.length ? (intersectionExtractor = _, upset) : intersectionExtractor; };\n  upset.elementExtractor = function(_) { return arguments.length ? (elementExtractor = _, upset) : elementExtractor; };\n  upset.setKeySortingFunction = function(_) { return arguments.length ? (setKeySortingFunction = _, upset) : setKeySortingFunction; };\n  upset.intersectionKeySortingFunction = function(_) { return arguments.length ? (intersectionKeySortingFunction = _, upset) : intersectionKeySortingFunction; };\n\n  upset.yObjectSize = function(_) { return arguments.length ? (yObjectSize = _, upset) : yObjectSize; };\n  upset.ySpacerSize = function(_) { return arguments.length ? (ySpacerSize = _, upset) : ySpacerSize; };\n  upset.xObjectSize = function(_) { return arguments.length ? (xObjectSize = _, upset) : xObjectSize; };\n  upset.xSpacerSize = function(_) { return arguments.length ? (xSpacerSize = _, upset) : xSpacerSize; };\n\n  function upset() {\n    // for convenience in handling orientation specific values\n    var horizontalQ = (orient == 'horizontal') ? true : false\n    var verticalQ = !horizontalQ\n\n    // background cliping rectangle\n    var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}\n    var container = setupContainer( selection, namespace, bgcpRect, backgroundFill );\n\n\n    cellKeys = d3.keys(data)\n    setValues = unique(cellKeys.map(setExtractor)).sort()\n    intersectionValues = unique(cellKeys.map(intersectionExtractor)).sort().sort(function(a, b){\n      return a.split(';').length - b.split(';').length\n    })\n\n    if (!horizontalQ) {\n      cellKeys.sort(function(a, b){ return setKeySortingFunction(a, b) || intersectionKeySortingFunction(a, b) })\n    } else {\n      cellKeys.sort(function(a, b){ return intersectionKeySortingFunction(a, b) || setKeySortingFunction(a, b) })\n    }\n\n\n\n\n    var\n    xValues = horizontalQ ? intersectionValues : setValues,\n    yValues = horizontalQ ? setValues : intersectionValues,\n    xDim = horizontalQ ? xValues.length : yValues.length,\n    yDim = horizontalQ ? yValues.length : xValues.length\n\n    // console.log(xValues, yValues)\n\n\n    xObjectSize = calculateWidthOfObject(spaceX, xDim, minObjectSize, maxObjectSize, xObjectSpacer, overflowQ)\n    yObjectSize = calculateWidthOfObject(spaceY, yDim, minObjectSize, maxObjectSize, yObjectSpacer, overflowQ)\n    xSpacerSize = calculateWidthOfSpacer(xValues, spaceX, xObjectSize, xDim, xObjectSpacer, overflowQ)\n    ySpacerSize = calculateWidthOfSpacer(yValues, spaceY, yObjectSize, yDim, yObjectSpacer, overflowQ)\n\n    var ySpacer = groupingSpacer()\n    .horizontalQ(false)\n    .moveby('category').numberOfObjects(yDim)\n    .objectSize(yObjectSize).spacerSize(ySpacerSize)\n    .transitionDuration(transitionDuration).easeFunc(easeFunc)\n\n    var xSpacer = groupingSpacer()\n    .horizontalQ(true)\n    .moveby('category').numberOfObjects(xDim)\n    .objectClass(objectClass)\n    .objectSize(xObjectSize).spacerSize(xSpacerSize)\n    .transitionDuration(transitionDuration).easeFunc(easeFunc)\n\n\n\n    if (verticalQ) {\n      xSpacer.objectClass(objectClass)\n      ySpacer.namespace('across').objectClass(hypenate(objectClass, 'across'))\n\n      ySpacer(container, yValues, 0)\n      container.selectAll('g.'+hypenate(objectClass, 'across'))\n      .each(function(d, i){ xSpacer(d3.select(this), xValues, 0) })\n    } else {\n      xSpacer.namespace('across').objectClass(hypenate(objectClass, 'across'))\n      ySpacer.objectClass(objectClass)\n\n      xSpacer(container, xValues, 0)\n      container.selectAll('g.'+hypenate(objectClass, 'across'))\n      .each(function(d, i){ ySpacer(d3.select(this), yValues, 0) })\n    }\n\n\n    var cells = container.selectAll('g:not(.to-remove).'+objectClass)\n    var lookup = {}\n    cellKeys.map(function(k, i){\n      lookup[setExtractor(k)+'::'+intersectionExtractor(k)] = k\n    })\n\n    // var positionedCellKeys = []\n    // for (var i = 0; i < setValues.length; i++) {\n    //   for (var j = 0; j < intersectionValues.length; j++) {\n    //     var lookupValue = lookup[setValues[j]+\"::\"+intersectionValues[i]]\n    //     if (lookupValue == undefined) {\n    //       positionedCellKeys.push(undefined)\n    //     } else {\n    //       positionedCellKeys.push(lookupValue)\n    //     }\n    //     console.log(i, j, lookupValue)\n    //   }\n    // }\n    //\n    // // console.log(positionedCellKeys)\n    //\n    // cells.data(positionedCellKeys);\n\n\n    cells.data(cellKeys);\n\n    cells.each(function(key, i) {\n      var t = d3.select(this)\n      if (key == undefined) {return }\n      var\n      currentData = data[key]\n      // console.log(key, currentData)\n      var\n      set = setExtractor(key, i),\n      intersection = intersectionExtractor(key, i)\n\n      // console.log(set, intersection)\n\n      t.classed(intersection, true)\n      t.classed(set, true)\n\n      var c = safeSelect(t, 'circle', hypenate(objectClass,'circle'))\n      c.attr('cx', xObjectSize / 2)\n      .attr('cy', yObjectSize / 2 )\n      .attr('r', radius == undefined ? Math.min(xObjectSize, yObjectSize) / 2 : radius)\n      .attr('fill', intersection.includes(set) ? \"black\": 'rgb(233,233,233)')\n      .attr('stroke', \"black\")\n      .attr(\"in-intersection\", intersection.includes(set))\n    })\n\n\n\n  }\n\n  function intersectionTotals() {\n    var totals = {}\n    // intersectionValues.sort(function(a, b){ return intersectionKeySortingFunction(a, b) })\n\n    intersectionValues.map(function(k, i){ totals[k] = {'total': 0} })\n    cellKeys.map(function(k, i){\n      var e = elementExtractor(k, i);\n      if (totals[intersectionExtractor(k, i)]['total'] == 0) {\n        if (Array.isArray(e)) {\n          totals[intersectionExtractor(k, i)]['total']+= e.length\n          totals[intersectionExtractor(k, i)]['values'] = e\n        } else {\n          totals[intersectionExtractor(k, i)]['total']+= e\n        }\n\n      }\n    })\n    return totals\n  }\n\n  function setTotals(){\n    var totals = {}\n    // intersectionValues.sort(function(a, b){ return intersectionKeySortingFunction(a, b) })\n\n    setValues.map(function(k, i){ totals[k] = {'total': 0} })\n\n    cellKeys.map(function(k, i){\n      var e = elementExtractor(k, i);\n      if (Array.isArray(e)) {\n        totals[setExtractor(k, i)]['total']+= e.length\n      } else {\n        totals[setExtractor(k, i)]['total']+= e\n      }\n    })\n    return totals\n  }\n\n  upset.intersectionTotals = intersectionTotals\n  upset.setTotals = setTotals\n\n  return upset\n}\n","import {hypenate, safeSelect} from './helpers';\nexport function filterTable(selection) {\n  var\n  data,\n  namespace = 'd3sm-filter-table',\n  sortQ = false,\n  fieldFunction = function(record, columnKey, recordValue) {return recordValue}\n\n  filterTable.data = function(_) { return arguments.length ? (data = _, filterTable) : data}\n  filterTable.namespace = function(_) { return arguments.length ? (namespace = _, filterTable) : namespace}\n  filterTable.fieldFunction = function(_) { return arguments.length ? (fieldFunction = _, filterTable) : fieldFunction}\n\n  function filterTable () {\n\n    var\n    container = safeSelect(selection, 'div', 'filter-table').classed(hypenate(namespace,'container'),true),\n\n    inputGroup = safeSelect(container, 'div', 'filter-input-group').classed('input-group',true),\n      inputPrepend = safeSelect(inputGroup, 'div', 'input-group-prepend'),\n        inputPrependSpan = safeSelect(inputPrepend, 'span', 'input-group-text').classed('search-button', true),\n          inputPrependSpanIcon = safeSelect(inputPrependSpan,'i','fa fa-search'),\n\n      input = safeSelect(inputGroup, 'input', 'form-control').attr('placeholder', 'filter').attr('type', 'text'),\n      inputAppend = safeSelect(inputGroup, 'div', 'input-group-append'),\n        inputAppendButton = safeSelect(inputAppend, 'a', 'close-button').classed('btn btn-outline-secondary', true),\n          inputAppendButtonIcon = safeSelect(inputAppendButton, 'i', 'fa fa-close'),\n\n\n    closeButton = inputAppendButton,\n\n    rTable = safeSelect(container, 'div', 'table-responsive'),\n      table = safeSelect(rTable, 'table', 'table')\n      .classed('table-hover', true)\n      .classed('table-bordered', true)\n      .classed(\"table-striped\", true),\n        tHead = safeSelect(table, 'thead')\n        .classed(\"thead-dark\", true),\n          header = safeSelect(tHead, 'tr'),\n        tBody = safeSelect(table, 'tbody')\n\n\n    var\n    rows = d3.keys(data),\n    cols = d3.keys(data[rows[0]])\n\n    var th = makeColumns(header, cols)\n    var tr = makeRows(tBody, rows)\n    var tr = populateRecords(tr, cols)\n\n\n\n    input.on('input', function(d, i){\n      var\n      val = input.property('value'),\n      reg = new RegExp(val, 'gi'),\n      use\n      if (val == '') {use = rows} else {\n        use = []\n        rows.map(function(r, i){\n          var row = data[r]\n          var fieldJoin = cols.map(function(c, j){\n            return fieldFunction(row, c, row[c])\n            // return row[c]\n          }).join('')\n          var match = fieldJoin.match(reg)\n          if (match == null || match.join('') == '') {}\n          else { use.push(r) }\n        })\n      }\n\n      tr = makeRows(tBody, use)\n      tr = populateRecords(tr, cols)\n    })\n\n    closeButton.on('click', function(d, i){\n      input.property('value', '').dispatch('input')\n    })\n\n  }\n\n  function makeColumns(header, cols) {\n    var th = header.selectAll('th')\n    th = th.data(cols)\n    th.exit().remove()\n    th = th.merge(th.enter().append('th'))\n    th.attr('scope', 'col').text(function(d, i){return d})\n    return th\n  }\n\n  function makeRows(body, rows) {\n    var tr = body.selectAll('tr')\n    tr = tr.data(rows)\n    tr.exit().remove()\n    tr = tr.merge(tr.enter().append('tr'))\n    return tr\n  }\n\n  function populateRecords(rows, cols) {\n    rows.each(function(r, i){\n      var record = data[r],\n      t = d3.select(this)\n\n      var fields = t.selectAll('td')\n      fields = fields.data(cols)\n      fields.exit().remove()\n      fields = fields.merge(fields.enter().append('td'))\n\n      fields.attr('scope', function(f, j){ return j == 0 })\n      .html(function(f, j){ return fieldFunction(record, f, record[f]) })\n\n    })\n    return rows\n  }\n\n\n\n  return filterTable\n}\n"],"names":["uniqueElements","value","index","self","indexOf","getTranslation","transform","g","document","createElementNS","undefined","setAttributeNS","matrix","baseVal","consolidate","e","f","modifyHexidecimalColorLuminance","hex","lum","String","replace","length","c","i","rgb","parseInt","substr","Math","round","min","max","toString","quartiles","data","qKeys","q2","d3","median","lower","filter","x","upper","q1","q0","q3","q4","k0","k1","k2","k3","k4","obj","hypenate","Array","prototype","slice","call","arguments","join","number","precision","shift","reverseShift","numArray","split","getContainingSVG","element","parent","parentElement","tag","tagName","toLowerCase","safeSelect","sel","cls","clsStr","sSel","select","empty","append","classed","attr","tickRange","n","a","s","push","hasQ","array","item","includes","unique","flatten","flat","map","isArray","concat","whichBin","bins","j","consoleGroup","name","window","d3sm","debugQ","group","consoleGroupEnd","groupEnd","log","func","msg","table","setupContainer","selection","namespace","rect","fill","container","y","width","height","bgRect","defs","cpRect","raise","calculateWidthOfObject","freeSpace","numberOfObjects","minObjectWidth","maxObjectWidth","sizeOfSpacer","overflowQ","remainingSpace","objectWidth","calculateWidthOfSpacer","baseSpacerSize","spacersAtEachLevel","spacersNeededAtEachLevel","level","levelData","reduce","b","isNaN","whiskerPath","dir","w","h","per","o","hh","ww","p","groupingSpacer","scaleLinear","easeSin","cur","d","horizontalQ","outerWidth","current","currentNode","node","selectAll","transition","duration","transitionDuration","ease","easeFunc","remove","recursivelyPosition","currentSelection","enter","exit","merge","exitFunction","each","this","levelSpacer","spacerSize","move","currentElement","t","enterFunction","moveby","scale","toRemove","objectClass","objectSize","size","_","colorFunction","interpolateRgb","colors","k","v","category","interpolate","interpolation","domain","dataExtent","range","helperScale","match","key","type","hoverQ","opac","fillOpacity","strokeOpacity","colorBy","categories","modifyOpacity","valueExtractor","cat","categoryExtractor","tooltip","keys","values","header","on","mousemove","currentData","mouse","div","style","cardBody","tBody","text","tr","rowKey","rowIndex","bbox","getBoundingClientRect","innerWidth","scrollX","event","pageX","innerHeight","scrollY","pageY","selectFilter","selectionName","defaultValue","lastValue","selectAppendButton","inputGroup","input","inputAppendButton","options","closeButton","currentStyle","property","dispatch","use","val","reg","RegExp","option","currentOption","parseFloat","lasso","svg","chartContainer","chartOffset","objectsOffset","eventCatcher","xScale","path","line","yScale","curve","curveLinearClosed","instance","animationRate","opacity","dashArray","stroke","strokeWidth","lassoedStroke","lassoedStrokeWidth","easeExp","paths","pEnter","activeQ","objectContainer","allPoints","render","preventDefault","stopPropagation","addEventListener","drag","removeEventListener","currentPoints","applyPathAttributes","color","which","pt","invert","lastPt","p1","p2","sqrt","euclideanDistance","tickDistance","tick","detect","lassos","box","absolutePosition","left","top","right","bottom","boxPts","inAnyLassoQ","lassoPoints","every","polygonContains","coord","updateObjects","applyObjectAttributes","setQ","preLassoFill","preLassoStroke","preLassoStrokeWidth","lassoedFill","keyFrames","html","draw","toggle","state","thisSVG","absolute","elementPosition","svgPosition","relativePositionTo","axis","label","tickTickLabelSpacer","tickLabelMargin","reverseScaleQ","orient","verticalQ","bgcpRect","spaceX","spaceY","guideLinesQ","guidelineSpace","tickLabelMaxFontSize","backgroundFill","tickLabelTextAnchor","tickLabelRotation","tickData","categoricalQ","grouping","tickLabels","numberOfTicks","extent","tickValues","flatTickData","space","reverse","domainPadding","minObjectSize","maxObjectSize","objectSpacer","objClass","spacerFunction","moveXBy","moveYBy","mt","datum","dist","labelNameGroup","labelElement","that","tickLength","tickStroke","tickStrokeWidth","string","chars","roundTo","tickLabelFontSize","getComputedTextLength","labelHover","labelHoverOff","tickLabelOnClick","guideLineStroke","guideLineStrokeWidth","lineStroke","lineStrokeWidth","parentNode","tickLabelOnHoverFunc","gline","minorQ","ii","tickLabelMinFontSize","tickLabelFunc","bar","keyA","keyB","descending","CF","TTip","barPercent","barKeys","ordered","sort","sortingFunction","barValues","defaultExit","nodes","parentIndexArray","Number","fillColor","strokeColor","barStrokeWidth","bubbleHeatmap","xKey","yKey","rKey","vKey","xKeySortingFunction","xExtractor","yKeySortingFunction","yExtractor","bhm","cellKeys","rExtractor","vExtractor","xValues","yValues","xDim","yDim","rValues","ySize","xSize","ySpacer","ySpacerSize","xSpacer","xSpacerSize","cells","vValues","radius","scaled","bubbleStrokeWidth","heatmap","hm","yMinObjectSize","yMaxObjectSize","xMinObjectSize","xMaxObjectSize","lookup","positionedCellKeys","lookupValue","objectStrokeWidth","boxwhisker","quartilesKey","quartilesKeys","boxKeys","boxValues","whisk","uWhisk","lWhisk","quart","uQuart","lQuart","mQuart","boxStrokeWidth","r","dif","dd","whiskerWidthPercent","whiskerStrokeWidth","datatoggle","xAxisOptions","yAxisOptions","yAxisSelectQ","xAxisSelectQ","updateFunction","currentKeys","vals","filterSelects","dataopts","doEnter","sf","scatter","pointKeys","valueExtractorX","valueExtractorY","valueExtractorR","extentX","valuesX","domainPaddingX","extentY","valuesY","domainPaddingY","extentR","valuesR","domainPaddingR","minRadius","maxRadius","points","pExit","scaleX","scaleY","scaleR","pointStrokeWidth","plotZoom","chart","xAxis","yAxis","chartSel","xAxisSel","yAxisSel","setLocks","chartObjSel","chartObjTrans","cos","getBBox","xLock","yLock","zoom","chartBox","xAxisBox","yAxisBox","eventType","deltaY","wheelSpeed","shiftQ","shiftKey","applyX","applyY","xAxisObjSel","yAxisObjSel","reset","multiPlotZoom","xComponents","yComponents","xComponentsSel","yComponentsSel","xComponentObjSel","yComponentObjSel","violin","base","minMaxHexScale","scaledColor","mod","quartileKeys","pointsTooltip","violinKey","violinData","violinPointKey","violinPointData","calcValues","calculateViolinValues","violinPoints","violinPointsExtractor","violinPointsKeys","violinPointsValues","pk","violinPointValueExtractor","pointQuartiles","binned","histogram","frequencies","bin","minContourPoint","maxContourPoint","violinContourPoints","x0","x1","contour","pointValues","neededViolinValues","vk","violinKeys","frequencyMax","vScale","lArea","curveBasis","rArea","area","la","ra","quarts","pointsQ","ptsContainer","pts","ptsEnter","pMin","pMax","pointRadius","pointKey","random","pointColorFunc","createEvent","initEvent","dispatchEvent","tooltipKey","quartileKey","violinValues","numericLegend","fontSize","textColor","legend","linearGradient","m","categoricLegend","ascending","catKeys","cx","cy","lassoWidget","maxNumberOfGroups","dataExtractor","onError","submit","clickSend","removeBtnClickToRemove","tabs","panes","remainingTabs","lastTab","curText","populatePane","pane","lead","btns","cN","btn","lassoBtn","makeLassoButton","currentLasso","clearBtn","lassoBtnToggle","las","tableData","extracted","makeClearButton","monitorLassoButtonState","target","makeTable","headR","body","headerKeys","headerCols","updateTableHeaderColumns","bodyRows","updateTableRows","rowData","cols","splice","makeNewGroup","gs","newGroupNumber","li","insert","txt","insertTab","makeTabPane","card","tabList","tabContent","rightTabs","upset","setValues","intersectionValues","xObjectSize","circleStrokeWidth","setExtractor","intersectionExtractor","elementExtractor","elementValues","yObjectSpacer","setKeySortingFunction","intersectionKeySortingFunction","xObjectSpacer","yObjectSize","set","intersection","intersectionTotals","totals","total","setTotals","filterTable","sortQ","record","columnKey","recordValue","rows","th","makeColumns","populateRecords","makeRows","row","fieldFunction","fields","extractViolinValues","valueExtractorFunction","qKey","minPoint","maxPoint","interpolateColors","interpolateRgbBasis","truncateText","setupStandardChartContainers","margins","axes","leg","margin","pos","Height","svgSpace","margPx","chartSpace","axesSpace","legRect","drawingSpace","yAxisRect","plotRect","xAxisRect","myLog","warn","console","info","error","resizeDebounce","wait","resize","immediate","timeout","context","args","callNow","setTimeout","apply","debounce"],"mappings":"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"} diff --git a/build/js/d3sm.v0.0.3.js b/build/js/d3sm.v0.0.3.js deleted file mode 100644 index 7b31405..0000000 --- a/build/js/d3sm.v0.0.3.js +++ /dev/null @@ -1,10613 +0,0 @@ -var d3sm = (function (exports) { - 'use strict'; - - // import {hasQ} from './array-functions'; - /******************************************************************************* - ** ** - ** ** - ** HELPERS ** - ** ** - ** ** - *******************************************************************************/ - /** - * Helper function for Array.filter to get unique elements of the array - * @param {*} value current value as mapping over array (self) - * @param {number} index current index in the array - * @param {Array} self passed array from Array.filter method - * @returns {boolean} whether or not value is the first of its kind (i.e. indexOf(value) == index) - */ - function uniqueElements(value, index, self) { - return self.indexOf(value) === index; - } - - /** - * Extracts x and y of translate from transform property - * @param {string} transform transform property of svg element - * @returns {number[]} x, y of translate(x, y) - */ - function getTranslation(transform) { - // Create a dummy g for calculation purposes only. This will never - // be appended to the DOM and will be discarded once this function - // returns. - var g = document.createElementNS('http://www.w3.org/2000/svg', 'g'); - // Set the transform attribute to the provided string value. - transform = transform == undefined ? 'translate(0,0)' : transform; - g.setAttributeNS(null, 'transform', transform); - // consolidate the SVGTransformList containing all transformations - // to a single SVGTransform of type SVG_TRANSFORM_MATRIX and get - // its SVGMatrix. - var matrix = g.transform.baseVal.consolidate().matrix; - // As per definition values e and f are the ones for the translation. - return [matrix.e, matrix.f]; - } - - /** - * Modifies luminance of hexidecimal number - * @param {string} hex should be hexidecimal value with or without the proceeding octotrope - * @param {number} lum value to increase or decrease luminosity by - * @returns {string} updated hexidecimal value without the proceeding octotrope - */ - function modifyHexidecimalColorLuminance(hex, lum) { - // validate hex string - var hex = String(hex).replace(/[^0-9a-f]/gi, ''); - - if (hex.length < 6) { - hex = hex[0] + hex[0] + hex[1] + hex[1] + hex[2] + hex[2]; - } - lum = lum || 0; - - // convert to decimal and change luminosity - var rgb = '#', - c, - i; - for (i = 0; i < 3; i++) { - c = parseInt(hex.substr(i * 2, 2), 16); - c = Math.round(Math.min(Math.max(0, c + c * lum), 255)).toString(16); - rgb += ('00' + c).substr(c.length); - } - - return rgb; - } - - /** - * Calculated the quartiles of the passed data and stores them with qKeys - * @param {number[]} data list of numerical values - * @param {string[]} [qKeys=['q0', 'q1', 'q2', 'q3', 'q4']] how returned object with quartiles should be stored - * @returns {Object} with keys qKeys giving only the numerical values for the quartiles - */ - function quartiles(data, qKeys) { - var q2 = d3.median(data), - lower = data.filter(function (x) { - return x < q2; - }), - upper = data.filter(function (x) { - return x > q2; - }), - q1 = d3.median(lower), - q1 = q1 == undefined ? q2 : q1, - q0 = d3.min(lower), - q0 = q0 == undefined ? q1 : q0, - q3 = d3.median(upper), - q3 = q3 == undefined ? q2 : q3, - q4 = d3.max(upper), - q4 = q4 == undefined ? q3 : q4, - k0 = 'q0', - k1 = 'q1', - k2 = 'q2', - k3 = 'q3', - k4 = 'q4', - obj = {}; - if (qKeys != undefined && qKeys.length == 5) { - k0 = qKeys[0];k1 = qKeys[1];k2 = qKeys[2];k3 = qKeys[3];k4 = qKeys[4]; - } - obj[k0] = q0;obj[k1] = q1;obj[k2] = q2;obj[k3] = q3;obj[k4] = q4; - - return obj; - } - - /** - * Helper function to get all values needed in making violin plots - * @param {string[]} violinKeys - * @param {number[]} data - * @param {Function} valueExtractorFunction how to get values from data[violinKeys[i]] - * @param {boolean} horizontalQ whether or not violins will be rendered horizontally or vertically - * @param {string} qKey how the object containing the quartiles should be labeled as - * @param {string[]} qKeys how each quartile should be labeled as - * @returns {Object} required for @see{@link violin} containing keys values, binnned, frequencies, points, and quartiles - * @see{@link quartiles} - */ - function extractViolinValues(violinKeys, data, valueExtractorFunction, horizontalQ, qKey, qKeys) { - var obj = {}; - violinKeys.map(function (k, i) { - var d = valueExtractorFunction(k, i, data), - binned = d3.histogram()(d), - frequencies = binned.map(function (x) { - return x.length; - }), - minPoint = horizontalQ ? { y: d3.min(d), x: 0 } : { x: d3.min(d), y: 0 }, - maxPoint = horizontalQ ? { y: d3.max(d), x: 0 } : { x: d3.max(d), y: 0 }, - points = binned.map(function (bin, i) { - return horizontalQ ? { y: bin.length ? d3.median(bin) : d3.median([bin.x0, bin.x1]), x: frequencies[i] } : { x: bin.length ? d3.median(bin) : d3.median([bin.x0, bin.x1]), y: frequencies[i] }; - }), - quarts = quartiles(d, qKeys), - o = { - values: d, - binned: binned, - frequencies: frequencies, - points: [minPoint].concat(points).concat([maxPoint]) - }; - o[qKey] = quarts; - obj[k] = o; - }); - return obj; - } - - /** - * Hypenates all strings together - * @param {string[]} arguments - * @returns {string} "arg1-arg2-...-argn" - */ - function hypenate() { - return Array.prototype.slice.call(arguments).join('-'); - } - - /** - * Rounds decimals of number to precision - * @param {number} number - * @param {number} precision - * @returns {number} rounded to precision - */ - function round(number, precision) { - var shift = function shift(number, precision, reverseShift) { - if (reverseShift) { - precision = -precision; - } - var numArray = ('' + number).split('e'); - return +(numArray[0] + 'e' + (numArray[1] ? +numArray[1] + precision : precision)); - }; - return shift(Math.round(shift(number, precision, false)), precision, true); - } - - /** - * recursively ascends element.parentElement to find a svg tag - * @param {Element} element - * @returns {Element | undefined} - */ - function getContainingSVG(element) { - var parent = element.parentElement; - var tag = parent.tagName.toLowerCase(); - if (tag === 'svg') { - return parent; - } - if (tag === 'html') { - return undefined; - } - return getContainingSVG(parent); - } - - /** - * Maps arguments in to d3.interpolateRgbBasis - * @param arguments - * @returns {Function} - */ - function interpolateColors() { - return d3.interpolateRgbBasis(arguments); - } - - /** - * Trys to reduce text to fit in specified area, made for tick labels as called by - * @see{@link axis} - * @param {d3.selection} t container for specific axis tick - * @param {string} text to be the label of the passed axis tick - * @param {boolean} orient of the axis, true is horizontal, false is vertical - * @param {number} tickLength is the length of the text - * @param {number} space is the amount of availble space for the text and the tick to fit in - * @param {boolean} overflowQ whether or not allowed to go over the alloted space - * @returns {none} - */ - function truncateText(t, text, orient, tickLength, space, overflowQ) { - var rect = t.node().getBoundingClientRect(); - t.text(text); - while (Math.max(rect.width, rect.height) > space - tickLength) { - text = String(text); - text = text.slice(0, text.length - 1); - t.text(text + '...'); - rect = t.node().getBoundingClientRect(); - if (text.length == 0) break; - } - } - - function truncateString(string, space, font) { - var chars = space / font; - if (chars < string.length) { - return string.slice(0, Math.round(chars - 5)) + '...'; - } else { - return string; - } - } - - /** - * Trys to use d3.selection to get element, if it doesnt exist, makes one - * @param {d3.selection} sel selection in which to try and find object - * @param {string} tag tag of which to try and select - * @param {string} [cls=''] class of tag to try and grab - * @returns {d3.selection} of either append or selected tag.cls within sel - */ - function safeSelect(sel, tag, cls) { - var clsStr = cls == undefined ? '' : '.' + cls; - var sSel = sel.select(tag + clsStr).empty() ? sel.append(tag) : sel.select(tag + clsStr); - return sSel.classed(clsStr.replace('.', ''), true).attr('transform', sSel.attr('transform') == undefined ? 'translate(0,0)' : sSel.attr('transform')); - } - - /** - * evenly partitions the range [min, max] into n parts - * @param {number} min - * @param {number} max - * @param {number} n - * @returns {number[]} array of length n evenly partitioned between min and max - */ - function tickRange(min, max, n) { - var a = [min]; - var d = max - min; - var s = d / (n - 1); - for (var i = 0; i < n - 2; i++) { - a.push(min + s * (i + 1)); - } - a.push(max); - return a; - } - - function euclideanDistance(p1, p2) { - var a = p1[0] - p2[0], - b = p1[1] - p2[1]; - return Math.sqrt(a * a + b * b); - } - - /** - * Short-hand for array.includes(item); - * @param {Array} array - * @param {*} item to test if contained in {array} - * @returns {boolean} - */ - function hasQ(array, item) { - return array.includes(item); - } - - /** - * Calculates the total value of numbers in passed array - * @param {number[]} array of numerical values - * @returns {number} sum over elements in array - */ - function total(array) { - return array.reduce(function (a, b) { - return a + b; - }, 0); - } - /** - * Removes duplicates in array - * @param {Array} array of items - * @returns {Array} of items such that item_i != item_j for all i < j - * @see{@link uniqueElements} for the filtering function - */ - function unique(array) { - return array.filter(uniqueElements); - } - - /** - * Concats all nested arrays in passed array to form a single array - * @param {Array} array of putatively nested arrays - * @param {Array} [flat=[]] current flattened array - * @returns {Array} with every element in the same level - */ - function flatten(array, flat) { - flat = flat == undefined ? [] : flat; - array.map(function (e, i) { - if (Array.isArray(e)) { - flat = flat.concat(flatten(e)); - } else { - flat.push(e); - } - }); - return flat; - } - - /** - * Search of list of lists to find which - if any - passed value is in - * @param {Array[]} bins list of lists of values - * @param {*} value item to test if in any of the bins - * @returns {number} indicating the index of the bin in which value was found - */ - function whichBin(bins, value) { - var i = -1; - for (var j = 0; j < bins.length; j++) { - if (hasQ(bins[j], value)) { - return j; - } - } - return i; - } - - /** - * calls console.group if d3sm.debugQ == true - * @param {string} name of the group - * @returns {undefined} - */ - function consoleGroup(name) { - if (window.d3sm.debugQ === true) { - console.group(name); - } - } - - /** - * calls console.groupEnd if d3sm.debugQ == true - * @returns {undefined} - */ - function consoleGroupEnd() { - if (window.d3sm.debugQ === true) { - console.groupEnd(); - } - } - - /** - * Calls console.log if d3sm.debugQ == true - * @param {string} func name of the function logging - * @param {string} msg to log - * @param {Object} data to be logged along side the message - * @returns {undefined} - */ - function log(func, msg, data) { - if (window.d3sm.debugQ === true) { - console.log('%c[d3sm::' + func + ']:\t' + msg, ['background: #6cd1ef', 'border-radius: 5000px', 'padding: 0px 2px', 'font-size: 14px'].join(';')); - console.table(data); - // console.trace() - } - } - - /** - * Calls console.warn if d3sm.debugQ == true - * @param {string} func name of the function warning - * @param {string} msg to display - * @param {Object} data to be displayed along side the message - * @returns {undefined} - */ - function warn(func, msg, data) { - if (window.d3sm.debugQ === true) console.warn('%c[d3sm::' + func + ']:\t' + msg, ['background: #ffd53e', 'border-radius: 5000px', 'padding: 0px 2px', 'font-size: 14px'].join(';')); - console.table(data); - } - /** - * Calls the console.info if d3sm.debugQ == true - * @param {string} func name of the function providing info - * @param {string} msg to display - * @param {Object} data to be displayed along side the message - * @returns {undefined} - */ - function info(func, msg, data) { - if (window.d3sm.debugQ) console.info('%c[d3sm::' + func + ']:\t' + msg, ['background: #009ccd', 'border-radius: 5000px', 'padding: 0px 2px', 'font-size: 14px'].join(';')); - console.table(data); - } - - /** - * Calls console.error if d3sm.debugQ == true - * @param {string} func name of the function which sends the error - * @param {string} msg to display - * @param {Object} data to be displayed along side the message - * @returns {undefined} - */ - function error(func, msg, data) { - if (window.d3sm.debugQ) console.error('[d3sm::' + func + ']:\t' + msg + '\t%o', data); - } - - /** - * Function for setting up containers for most plots with the y axis container - * positioned on the left and the x axis container positioned on the bottom - * @param {d3.selection} selection selection of container in which the svg is or should be made - * @param {string} namespace namespace of the chart - * @param {Object} [space={w:window.innerWidth, h:window.innerHeight}] the width (w) and height (h) availble - * @param {number} [space.w=window.innerWidth] the available width in which to render the chart - * @param {number} [space.h=window.innerHeight] the available height in which to render the chart - - * @param {Object} [margins={top: 0.01, bottom: 0.01, left: 0.01, right: 0.01}] the margins for the chart - * @param {number} [margins.top=0.01] the top margin of the chart - * @param {number} [margins.bottom=0.01] the bottom margin of the chart - * @param {number} [margins.left=0.01] the left margin of the chart - * @param {number} [margins.right=0.01] the right margin of the chart - - - * @param {Object} [percentages = {axes:{x:0.1,y:0.1},space:{w:0.8,h:0.6}}] percentages of the paramater space of which to make the x and y axes as well as the percent of the availble space in which to render the plot - * @param {Object} [percentages.axes={x:0.1,y:0.1}] the percentages of the paramater space, of which the x and y axes will take up - * @param {number} [percentages.axes.xAxisPercent=0.1] the percentages of the paramater space, of which the x axis will take up - * @param {number} [percentages.axes.yAxisPercent=0.1] the percentages of the paramater space, of which the y axis will take up - - * @param {Object} [percentages.space={w:0.8,h:0.6}] the percentages of the paramater space, of which the SVG's width and height will be set - * @param {number} [percentages.space.percentOfSpaceForWidth=0.1] the percentages of the paramater space, of which the SVG's width will be set - * @param {number} [percentages.space.percentOfSpaceForHeight=0.1] the percentages of the paramater space, of which the SVG's height will be set - - * @returns {Object} returns the selection and "boundingRects" of the plot container, x-axis container and y-axis container - * as - * - * { - * - * plot: {selection: plotSelection, rect: plotRect}, - * - * xAxis:{selection:xAxisSelection, rect:xAxisRect}, - * - * yAxis: {selection:yAxisSelection, rect:yAxisRect} - * - * } - * - * where each rect has form: - * - * {x: #, y: #, h: #, w: #} - * - * depicting the starting x and y coordinate of the coresponding container (also their default transform values) as well their height (h) ans width (w) - */ - // export function setupStandardChartContainers( selection, namespace, space, margins, percentages) { - // export function setupStandardChartContainers( - // selection, - // namespace, - // space={w:availableWidth=window.innerWidth, h:availableHeight=window.innerHeight}, - // margins={top:0.01, bottom:0.01, left:0.01, right:0.01}, - // percentages={axes: {x: xAxisPercent=0.1, y: yAxisPercent=0.1}, space: {w: percentOfSpaceForWidth, h: percentOfSpaceForHeight}} - // ) { - // if (space == undefined) { space = {w: window.innerWidth, h: window.innerHeight} } - // if (margins == undefined) { margins = {top: 0.01, bottom: 0.01, left: 0.01, right: 0.01} } - // if (percentages == undefined) { percentages = {}; } - // if (percentages.axes == undefined) { percentages.axes = { x:0.1, y:0.1 } } - // if (percentages.space == undefined) { percentages.space = { w: 0.8, h: 0.6 } } - // - // // SVG width and height - // var svgSpace = { - // w: space.w * percentages.space.w, - // h: space.h * percentages.space.h - // }, - // - // // Space after removing margins - // chartSpace = { - // w: svgSpace.w - (margins.left * space.w) - (margins.right * space.w), - // h: svgSpace.h - (margins.top * space.h) - (margins.bottom * space.h) - // }, - // - // // main dimension of x and y axies - // // e.g. defines how tall x axis is as length is determined by plotRect.w - // axesSpace = { - // x: chartSpace.h * percentages.axes.x, - // y: chartSpace.w * percentages.axes.y - // }, - // - // // space left for drawing the chart properly (e.g. bars, violins, etc) - // drawingSpace = { - // x: chartSpace.w - axesSpace.y, - // y: chartSpace.h - axesSpace.x - // }, - // - // - // yAxisRect = { - // x: axesSpace.y + (margins.left * space.w), - // y: (margins.top * space.h), - // w: axesSpace.y, - // h: drawingSpace.y - // }, - // - // plotRect = { - // x: axesSpace.y + (margins.left * space.w), - // y: (margins.top * space.h), - // w: drawingSpace.x, - // h: drawingSpace.y - // }, - // - // xAxisRect = { - // x: axesSpace.y + (margins.left * space.w), - // y: (margins.top * space.h + plotRect.h), - // w: drawingSpace.x, - // h: axesSpace.x - // } - // - // - // var container = safeSelect(selection, 'svg', namespace) - // .style('width', svgSpace.w+'px') - // .style('height', svgSpace.h+'px') - // - // var axes = safeSelect(container, 'g', hypenate(namespace, 'axes')) - // - // // .attr('transform', "translate("+plotRect.x+","+plotRect.y+")"), - // - // var plot = safeSelect(container, 'g', hypenate(namespace, 'plot')) - // .attr('transform', "translate("+plotRect.x+","+plotRect.y+")") - // - // var xAxis = safeSelect(axes, 'g', hypenate(namespace, 'x-axis')) - // .attr('transform', "translate("+xAxisRect.x+","+xAxisRect.y+")") - // - // var yAxis = safeSelect(axes, 'g', hypenate(namespace, 'y-axis')) - // .attr('transform', "translate("+yAxisRect.x+","+yAxisRect.y+")") - // - // return { - // svg: { - // selection: container, - // rect: svgSpace - // }, - // plot: { - // selection: plot, - // rect: plotRect - // }, - // xAxis: { - // selection: xAxis, - // rect: xAxisRect - // }, - // yAxis: { - // selection: yAxis, - // rect: yAxisRect - // } - // } - // - // // return [plot, xAxis, yAxis] - // } - // - // - - - function setupStandardChartContainers(selection, namespace, container) { - var margins = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : { top: 0.01, bottom: 0.01, left: 0.01, right: 0.01 }; - var svg = arguments.length > 4 && arguments[4] !== undefined ? arguments[4] : { w: 0.8, h: 0.6 }; - var axes = arguments.length > 5 && arguments[5] !== undefined ? arguments[5] : { y: 0.1, x: 0.1 }; - var leg = arguments.length > 6 && arguments[6] !== undefined ? arguments[6] : { x: 0, margin: 0, pos: 'left' // absolute width of legend and space on either size - }; - - if (container == undefined) { - container = { w: window.innerWidth, h: window.Height }; - } - // SVG width and height - - var svgSpace = { - w: container.w * svg.w, - h: container.h * svg.h - }; - - var margPx = { - top: margins.top * svgSpace.h, - bottom: margins.bottom * svgSpace.h, - left: margins.left * svgSpace.w, - right: margins.right * svgSpace.w - }, - - - // Space after removing margins - chartSpace = { - w: svgSpace.w - margPx.left - margPx.right, - h: svgSpace.h - margPx.top - margPx.bottom - }, - - - // main dimension of x and y axies - // e.g. defines how tall x axis is as length is determined by plotRect.w - axesSpace = { - x: chartSpace.h * axes.x, - y: chartSpace.w * axes.y - }, - - - // space left for drawing the chart properly (e.g. bars, violins, etc) - drawingSpace = { - x: chartSpace.w - axesSpace.y - leg.x - 2 * leg.margin, - y: chartSpace.h - axesSpace.x - }, - legRect = { - x: leg.margin + margPx.left + (leg.pos == 'left' ? 0 : drawingSpace.x + axesSpace.y), - y: margPx.top, // this is soomehow getting calculated incorectly - w: leg.x, - h: drawingSpace.y - }, - yAxisRect = { - x: axesSpace.y + margPx.left + (leg.pos == 'left' ? leg.x + 2 * leg.margin : 0), - y: margPx.top, - w: axesSpace.y, - h: drawingSpace.y - }, - plotRect = { - x: axesSpace.y + margPx.left + (leg.pos == 'left' ? leg.x + 2 * leg.margin : 0), - y: margPx.top, - w: drawingSpace.x, - h: drawingSpace.y - }, - xAxisRect = { - x: axesSpace.y + margPx.left + (leg.pos == 'left' ? leg.x + 2 * leg.margin : 0), - y: margPx.top + drawingSpace.y, - w: drawingSpace.x, - h: axesSpace.x - }; - - container = d3sm.safeSelect(selection, 'svg', namespace).style('width', svgSpace.w + 'px').style('height', svgSpace.h + 'px'); - - var axes = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'axes')); - - var leg = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'legend')).attr('transform', "translate(" + legRect.x + "," + legRect.y + ")"); - - var plot = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'plot')).attr('transform', "translate(" + plotRect.x + "," + plotRect.y + ")"); - - var xAxis = d3sm.safeSelect(axes, 'g', d3sm.hypenate(namespace, 'x-axis')).attr('transform', "translate(" + xAxisRect.x + "," + xAxisRect.y + ")"); - - var yAxis = d3sm.safeSelect(axes, 'g', d3sm.hypenate(namespace, 'y-axis')).attr('transform', "translate(" + yAxisRect.x + "," + yAxisRect.y + ")"); - - return { - svg: { - selection: container, - rect: svgSpace - }, - plot: { - selection: plot, - rect: plotRect - }, - xAxis: { - selection: xAxis, - rect: xAxisRect - }, - yAxis: { - selection: yAxis, - rect: yAxisRect - }, - legend: { - selection: leg, - rect: legRect - } - }; - } - - /** - * Adds a clip-path rect and binds it to container - * @param {d3.selection} container in which to add the clip-path and to which to bind the cliping path to - * @param {Object} rect the coordinates (x, y, width, height) of the clip-path - * @param {string} namespace - * @returns {d3.selection} of the clip-path rect - */ - function cpRect(container, rect, namespace) { - var defs = safeSelect(container, 'defs', hypenate(namespace, 'definitions')); - var cp = safeSelect(defs, 'clipPath', hypenate(namespace, 'clip-path')).attr('id', hypenate(namespace, 'clip-path')); - - var cpRect = safeSelect(cp, 'rect').attr('x', rect.x).attr('y', rect.y).attr('width', rect.width).attr('height', rect.height); - - defs.raise(); - // set clipping path to container - container.attr('clip-path', 'url(#' + hypenate(namespace, 'clip-path') + ')'); - - return cpRect; - } - - /** - * Adds a background rect t to container - * @param {d3.selection} container in which to add the background rectangle - * @param {Object} rect the coordinates (x, y, width, height) of the background - * @param {string} fill the color of the background - * @returns {d3.selection} of the background fill - */ - function bgRect(container, rect, fill) { - return safeSelect(container, 'rect', 'bg').attr('x', rect.x).attr('y', rect.y).attr('width', rect.width).attr('height', rect.height).attr('fill', fill); - } - - /** - * Sets up the container for making chart elements. This includes making - * a clip-path rect bound to the passed container, a background rect, and - * a g element with class -object-container. - * @param {d3.selection} container in which to add the clip-path and background - * @param {string} namespace - * @param {Object} rect the coordinates (x, y, width, height) of the background and clip-path - * @param {string} fill the color of the background - * @returns {d3.selection} of g.-object-container - * - * @see{@link bgRect} - * @see{@link cpRect} - */ - function setupContainer(selection, namespace, rect, fill) { - // the container for three main items, bg, defs, and object-container - var container = safeSelect(selection, 'g', namespace), - bg = bgRect(container, rect, fill), - cp = cpRect(container, rect, namespace), - objectContainer = safeSelect(container, 'g', hypenate(namespace, 'object-container')); - return objectContainer; - } - - /** - * determines the width of an object for the calling plotting function - * @param {number} freeSpace how much space is avalible - * @param {number} numberOfObjects how many object do we need - * @param {number} minObjectWidth how small are these objects allowed to be - * @param {number} maxObjectWidth how large are these object allowed to be - * @param {number} sizeOfSpacer percent of freeSpace that a single spacer should take up (need numberOfObjects - 1 spacers) - * @param {boolean} overflowQ can we go beyond alloted space - * @returns {number} how large object should be - * function tries to keep object within min / max width, but wil default to - * 5e-10 (smallest consistenly visible by svg size of element) if overflowQ is false - */ - function calculateWidthOfObject(freeSpace, numberOfObjects, minObjectWidth, maxObjectWidth, sizeOfSpacer, overflowQ) { - var sizeOfSpacer = sizeOfSpacer == 0 || sizeOfSpacer > 1 ? sizeOfSpacer : freeSpace * sizeOfSpacer; - - var numberOfSpacers = numberOfObjects - 1; - var spaceTakenBySpacers = numberOfSpacers * sizeOfSpacer; - var remainingSpace = freeSpace - spaceTakenBySpacers; - remainingSpace = remainingSpace < 0 ? 0 : remainingSpace; - var objectWidth = remainingSpace / numberOfObjects; - - if (overflowQ && minObjectWidth != undefined && objectWidth < minObjectWidth) { - objectWidth = minObjectWidth; - } - // if ( maxObjectWidth != undefined && objectWidth > maxObjectWidth ) { objectWidth = maxObjectWidth } - if (overflowQ && maxObjectWidth != undefined && objectWidth < maxObjectWidth) { - objectWidth = maxObjectWidth; - } - return Math.max(objectWidth, 5e-10); - } - - /** - * @param {Array[]} data list data (can be nested). If nested will create more complex spacer size - * @param {number} freeSpace how much space is avalible - * @param {number} objectWidth @see{@link calculateWidthOfObject} - * @param {number} numberOfObjects how many object do we need - * @param {number} baseSpacerSize percent of freeSpace that a single spacer should take up (need numberOfObjects - 1 spacers) - * @param {boolean} overflowQ can we go beyond alloted space - * @returns {number} returns size that spacer should be at level=0 - */ - function calculateWidthOfSpacer(data, freeSpace, objectWidth, numberOfObjects, baseSpacerSize, overflowQ) { - if (overflowQ) { - // var limitedNumberOfObjects = numberOfObjects > 6 ? 6 : numberOfObjects - // var spaceLeft = freeSpace - limitedNumberOfObjects * objectWidth - // return spaceLeft / (limitedNumberOfObjects - 1) - return freeSpace * baseSpacerSize; - } - var spacersAtEachLevel = spacersNeededAtEachLevel(data); - var totalSpacerPercent = total(spacersAtEachLevel.map(function (e, i) { - return e * 1 / (i + 1); - })); - var baseSpacerSize = (freeSpace - objectWidth * numberOfObjects) / totalSpacerPercent; - // console.log(freeSpace, objectWidth, numberOfObjects, totalSpacerPercent) - // console.log(totalSpacerPercent, baseSpacerSize, totalSpacerPercent * baseSpacerSize) - return isNaN(baseSpacerSize) ? 0 : baseSpacerSize; - } - - /** - * Calculates number of spacers needed to seperate elements at each level. - * @param {Array[]} array list data (can be nested). If nested will create more complex spacer size - * @param {number} [level=0] current level, used in recusrion - * @param {Array} [levelData=[]] how many spacers needed at a given level - * @returns {Array} levelData - * - * @example - * array = [[1,2], [3,4]] - * // returns [1, 2] - * as at level=0 the only spacer needed is between [1,2] and [3,4] - * and at level=1 the only two spacers needed is between 1 and 2 as well as - * 3 and 4 since the spacer between 2 and 3 is handled at level=0 - */ - function spacersNeededAtEachLevel(array, level, levelData) { - if (level == undefined) { - level = 0; - } else { - level += 1; - } - if (levelData == undefined) { - levelData = []; - } - if (level >= levelData.length) { - levelData.push(array.length - 1); - } else { - levelData[level] += array.length - 1; - } - array.map(function (e, i) { - if (Array.isArray(e)) { - spacersNeededAtEachLevel(e, level, levelData); - } - }); - return levelData; - } - - /** - * Draws a whisker for @see{@link boxwhisker} - * @param {boolean} dir direction to draw whisker, should be either true (up, top) or false (down or bottom) - * @param {number} x starting x coordinate in which to draw whisker - * @param {number} y starting y coordinate in which to draw whisker - * @param {number} w width of space in which to draw whisker - * @param {number} h height of space in which to draw whisker - * @param {number} per percentage of w or h (depends on o) to make whisker - * @param {boolean} o orientation, true is horizontal and false is vertical - * @returns {string} representing the svg path (i.e. the d attribute for a path tag) - */ - function whiskerPath(dir, x, y, w, h, per, o) { - // d = direction (true is up), p = percent width - if (dir == 'up' || dir == 'top' || dir == true) { - dir = true; - } - if (dir == 'down' || dir == 'bottom' || dir == false) { - dir = false; - } - o = o == undefined ? 'horizontal' : o; - per = per == undefined ? 1 : per; - if (o != "horizontal") { - var hh = h * per, - w = dir ? w : -w, - a = dir ? x + w : x, - b = dir ? x : x + w, - c = dir ? a : b; - p = "M " + a + ' ' + h / 2 + ' ' + 'L ' + b + ' ' + h / 2 + ' ' + 'M ' + c + ' ' + (h / 2 - hh / 2) + ' ' + 'L ' + c + ' ' + (h / 2 + hh / 2) + ' '; - - return p; - } - var ww = w * per, - a = dir ? y + h : y, - b = dir ? y : y + h, - p = "M " + w / 2 + ' ' + a + ' ' // straight line part - + 'L ' + w / 2 + ' ' + b + ' ' // straight line part - + 'h ' + -ww / 2 + ' ' + 0 + ' ' // horizontal line part - + 'h ' + ww + ' ' + 0 + ' '; - return p; - } - - function resizeDebounce(f, wait) { - var resize = debounce(function () { - f(); - }, wait); - window.addEventListener('resize', resize); - } - - function debounce(func, wait, immediate) { - var timeout; - return function () { - var context = this, - args = arguments; - var later = function later() { - timeout = null; - if (!immediate) func.apply(context, args); - }; - var callNow = immediate && !timeout; - clearTimeout(timeout); - timeout = setTimeout(later, wait); - if (callNow) func.apply(context, args); - }; - } - - /******************************************************************************* - ** ** - ** ** - ** SPACEGROUPING ** - ** ** - ** ** - *******************************************************************************/ - /** - * Produces a function for spacing objects by an arbitrarly complex grouping - * @returns {recursivelyPosition} the function for moving the objects - * (see {@link groupingSpacer#recursivelyPosition}) - * @namespace groupingSpacer - */ - function groupingSpacer() { - var - /*@var {boolean} horizontalQ @default*/ - - /** - * Whether or not to space objects horizontally or vertically. - * (see {@link groupingSpacer.horizontalQ}) - * @param {boolean} [horizontalQ=true] - * @memberof groupingSpacer# - * @instance - */ - horizontalQ = true, - - /** - * The scale to use to position elements if {@link groupingSpacer#moveby}="string" - * (see {@link groupingSpacer.scale}) - * @param {d3.scale} [scale=d3.scaleLinear()] - * @memberof groupingSpacer# - * @instance - */ - scale = d3.scaleLinear(), - - /** - * How elements in the complex grouping should be moved over by. - * By default, moveby="category", which moves objects by the complex grouping - * But objects can also be moved over by scale. - * (see {@link groupingSpacer.moveby}) - * @param {string} [moveby="category"] - * @memberof groupingSpacer# - * @instance - */ - moveby = 'category', - - /** - * How many objects are there in total - * (see {@link groupingSpacer.numberOfObjects}) - * @param {number} [numberOfObjects=none] - * @memberof groupingSpacer# - * @instance - */ - numberOfObjects, - - /** - * The class given to an nested tag whose parent(s) have the correct transition - * properties - * (see {@link groupingSpacer.numberOfObjects}) - * @param {string} [numberOfObjects='d3sm-groupped-item'] - * @memberof groupingSpacer# - * @instance - */ - objectClass = 'd3sm-groupped-item', - - /** - * The size of the objects being positioned - * (see {@link groupingSpacer.objectSize}) - * @param {number} [objectSize=none] - * @memberof groupingSpacer# - * @instance - */ - objectSize, - - /** - * The size of the un-nested spacer between objects - * (see {@link groupingSpacer.spacerSize}) - * @param {number} [spacerSize=none] - * @memberof groupingSpacer# - * @instance - */ - spacerSize, - - /** - * The duration of transitions in ms - * (see {@link groupingSpacer.transitionDuration}) - * @param {number} [transitionDuration=1000] - * @memberof groupingSpacer# - * @instance - */ - transitionDuration = 1000, - - /** - * The ease function for the transitions - * (see {@link groupingSpacer.easeFunc}) - * @param {d3.ease} [easeFunc=d3.easeSin] - * @memberof groupingSpacer# - * @instance - */ - easeFunc = d3.easeSin, - - /** - * The namespace for the objects being moved - * (see {@link groupingSpacer.namespace}) - * @param {string} [namespace='spacer'] - * @memberof groupingSpacer# - * @instance - */ - namespace = 'spacer', - - /** - * The animation for new objects being added - * (see {@link groupingSpacer.enterFunction}) - * @param {function} enterFunction - * @memberof groupingSpacer# - * @instance - * @example - * // by default - * function(newObjectSelection) { - * newObjectSelection.attr('transform', function(d, i){ - * var - * x = horizontalQ ? objectSize * numberOfObjects + spacerSize * (numberOfObjects - 1) : 0, - * y = !horizontalQ ? objectSize * numberOfObjects + spacerSize * (numberOfObjects - 1) : 0, - * t = 'translate('+x+','+y+')' - * return t - * }) - * } - */ - enterFunction = function enterFunction(cur) { - cur.attr('transform', function (d, i) { - var - // x = horizontalQ ? objectSize * numberOfObjects + spacerSize * (numberOfObjects - 1) : 0, - // y = !horizontalQ ? objectSize * numberOfObjects + spacerSize * (numberOfObjects - 1) : 0, - x = horizontalQ ? window.outerWidth : 0, - y = !horizontalQ ? window.outerWidth : 0, - t = 'translate(' + x + ',' + y + ')'; - // if(y == undefined) {console.log(cur.node(), y, d)} - return t; - }); - }, - - /** - * The animation for old objects being removed - * (see {@link groupingSpacer.exitFunction}) - * @param {function} exitFunction - * @memberof groupingSpacer# - * @instance - * @example - * // by default - * oldObjectSelection.transition().duration(transitionDuration).ease(easeFunc) - * .attr('transform', function(d, i){ - * var - * x = horizontalQ ? objectSize * numberOfObjects + spacerSize * (numberOfObjects - 1) : 0, - * y = !horizontalQ ? objectSize * numberOfObjects + spacerSize * (numberOfObjects - 1) : 0, - * t = 'translate('+x+','+y+')' - * return t - * }).remove() - */ - exitFunction = function exitFunction(cur) { - log("groupingSpacer", "exiting with", { current: cur, currentNode: cur.node() }); - cur.selectAll('g').classed('to-remove', true); - - cur.transition().duration(transitionDuration * 0.9).ease(easeFunc).attr('transform', function (d, i) { - var - // x = horizontalQ ? objectSize * numberOfObjects + spacerSize * (numberOfObjects - 1) : 0, - // y = !horizontalQ ? objectSize * numberOfObjects + spacerSize * (numberOfObjects - 1) : 0, - x = horizontalQ ? window.outerWidth : 0, - y = !horizontalQ ? window.outerWidth : 0, - t = 'translate(' + x + ',' + y + ')'; - // if(y == undefined) {console.log(cur.node(), y, d)} - return t; - }).remove(); - }; - - /** - * Gets / sets horizontalQ (whether or not to space objects horizontally or vertically). - * (see {@link groupingSpacer#horizontalQ}) - * @param {string} [_=none] - * @returns {groupingSpacer | string} - * @memberof groupingSpacer - * @static - */ - recursivelyPosition.horizontalQ = function (_) { - return arguments.length ? (horizontalQ = _, recursivelyPosition) : horizontalQ; - }; - /** - * Gets / sets the scale to use to position elements if {@link groupingSpacer#moveby}="string" - * (see {@link groupingSpacer#scale}) - * @param {d3.scale} [_=none] - * @returns {groupingSpacer | d3.scale} - * @memberof groupingSpacer - * @static - */ - recursivelyPosition.scale = function (_) { - return arguments.length ? (scale = _, recursivelyPosition) : scale; - }; - /** - * Gets / sets moveby (whether or not to move by scale or by grouping). - * (see {@link groupingSpacer#moveby}) - * @param {string} [_=none] - * @returns {groupingSpacer | string} - * @memberof groupingSpacer - * @static - */ - recursivelyPosition.moveby = function (_) { - return arguments.length ? (moveby = _, recursivelyPosition) : moveby; - }; - /** - * Gets / sets numberOfObjects. - * (see {@link groupingSpacer#numberOfObjects}) - * @param {number} [_=none] - * @returns {groupingSpacer | number} - * @memberof groupingSpacer - * @static - */ - recursivelyPosition.numberOfObjects = function (_) { - return arguments.length ? (numberOfObjects = _, recursivelyPosition) : numberOfObjects; - }; - /** - * Gets / sets the objectClass (will be applied to elements). - * (see {@link groupingSpacer#objectClass}) - * @param {string} [_=none] - * @returns {groupingSpacer | string} - * @memberof groupingSpacer - * @static - */ - recursivelyPosition.objectClass = function (_) { - return arguments.length ? (objectClass = _, recursivelyPosition) : objectClass; - }; - /** - * Gets / sets the objectSize. - * (see {@link groupingSpacer#objectSize}) - * @param {number} [_=none] - * @returns {groupingSpacer | number} - * @memberof groupingSpacer - * @static - */ - recursivelyPosition.objectSize = function (_) { - return arguments.length ? (objectSize = _, recursivelyPosition) : objectSize; - }; - /** - * Gets / sets the spacerSize. - * (see {@link groupingSpacer#spacerSize}) - * @param {number} [_=none] - * @returns {groupingSpacer | number} - * @memberof groupingSpacer - * @static - */ - recursivelyPosition.spacerSize = function (_) { - return arguments.length ? (spacerSize = _, recursivelyPosition) : spacerSize; - }; - /** - * Gets / sets the transitionDuration. - * (see {@link groupingSpacer#transitionDuration}) - * @param {number} [_=none] - * @returns {groupingSpacer | number} - * @memberof groupingSpacer - * @static - */ - recursivelyPosition.transitionDuration = function (_) { - return arguments.length ? (transitionDuration = _, recursivelyPosition) : transitionDuration; - }; - /** - * Gets / sets the easeFunc. - * (see {@link groupingSpacer#easeFunc}) - * @param {d3.ease} [_=none] - * @returns {groupingSpacer | d3.ease} - * @memberof groupingSpacer - * @static - */ - recursivelyPosition.easeFunc = function (_) { - return arguments.length ? (easeFunc = _, recursivelyPosition) : easeFunc; - }; - /** - * Gets / sets the namespace. - * (see {@link groupingSpacer#namespace}) - * @param {string} [_=none] - * @returns {groupingSpacer | string} - * @memberof groupingSpacer - * @static - */ - recursivelyPosition.namespace = function (_) { - return arguments.length ? (namespace = _, recursivelyPosition) : namespace; - }; - /** - * Gets / sets the enterFunction. - * (see {@link groupingSpacer#enterFunction}) - * @param {function} [_=none] - * @returns {groupingSpacer | function} - * @memberof groupingSpacer - * @static - */ - recursivelyPosition.enterFunction = function (_) { - return arguments.length ? (enterFunction = _, recursivelyPosition) : enterFunction; - }; - /** - * Gets / sets the exitFunction. - * (see {@link groupingSpacer#exitFunction}) - * @param {function} [_=none] - * @returns {groupingSpacer | function} - * @memberof groupingSpacer - * @static - */ - recursivelyPosition.exitFunction = function (_) { - return arguments.length ? (exitFunction = _, recursivelyPosition) : exitFunction; - }; - - /** - * recursively position the objects inside of the selection. - * @param {d3.selection} selection - * @param {Object} data - * @param {level} [level=0] recursion depth - * @returns {number} (how much to move next element) - * @memberof groupingSpacer# - */ - function recursivelyPosition(selection, data, level) { - if (level == undefined) { - level = 0; - } - - var currentSelection = selection.selectAll('g.' + namespace + '[level="' + level + '"]').data(data); - var enter = currentSelection.enter().append('g').attr('level', level).attr('class', namespace); - var exit = currentSelection.exit(); - currentSelection = currentSelection.merge(enter); - - if (typeof exitFunction == 'function') { - exit.each(function (d, i) { - exitFunction(d3.select(this)); - }); - } else { - exit.remove(); - } - // spacer for current level - var levelSpacer = spacerSize / (level + 1); - // movement for current level - var move = 0; - currentSelection.each(function (currentElement, index) { - var t = d3.select(this); - if (t.attr('transform') == undefined && typeof enterFunction == 'function') { - enterFunction(t); - } - - t.transition().duration(transitionDuration).ease(easeFunc).attr('transform', function (d, i) { - var x = horizontalQ ? moveby == "scale" ? scale(d) : move : 0, - y = !horizontalQ ? moveby == "scale" ? scale(d) : move : 0, - t = 'translate(' + x + ',' + y + ')'; - return t; - }); - - if (Array.isArray(currentElement)) { - move += recursivelyPosition(t, currentElement, level + 1); - var toRemove = t.selectAll('g.' + namespace + '[level="' + level + '"] > g.' + objectClass + '.' + namespace); - if (typeof exitFunction == 'function') { - toRemove.each(function (d, i) { - exitFunction(d3.select(this)); - }); - } else { - toRemove.remove(); - } - } else { - move += objectSize; - var obj = t.select('g.' + namespace + '[level="' + level + '"] > g.' + objectClass + '.' + namespace); - if (obj.empty()) { - obj = t.append('g').attr('class', objectClass).classed(namespace, true); - } - obj.attr('parent-index', index); - var toRemove = t.selectAll('g.' + namespace + '[level="' + (level + 1) + '"]'); - - if (typeof exitFunction == 'function') { - toRemove.each(function (d, i) { - exitFunction(d3.select(this)); - }); - } else { - toRemove.remove(); - } - } - move += index == currentSelection.size() - 1 ? 0 : levelSpacer; - }); - return move; - } - return recursivelyPosition; - } - - var slicedToArray = function () { - function sliceIterator(arr, i) { - var _arr = []; - var _n = true; - var _d = false; - var _e = undefined; - - try { - for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { - _arr.push(_s.value); - - if (i && _arr.length === i) break; - } - } catch (err) { - _d = true; - _e = err; - } finally { - try { - if (!_n && _i["return"]) _i["return"](); - } finally { - if (_d) throw _e; - } - } - - return _arr; - } - - return function (arr, i) { - if (Array.isArray(arr)) { - return arr; - } else if (Symbol.iterator in Object(arr)) { - return sliceIterator(arr, i); - } else { - throw new TypeError("Invalid attempt to destructure non-iterable instance"); - } - }; - }(); - - var toConsumableArray = function (arr) { - if (Array.isArray(arr)) { - for (var i = 0, arr2 = Array(arr.length); i < arr.length; i++) arr2[i] = arr[i]; - - return arr2; - } else { - return Array.from(arr); - } - }; - - /******************************************************************************* - ** ** - ** ** - ** AXIS ** - ** ** - ** ** - *******************************************************************************/ - - /** - * Creates an axis - * - * {@link https://sumneuron.gitlab.io/d3sm/demos/axes/index.html Demo} - * @constructor axis - * @param {d3.selection} selection - * @namespace axis - * @returns {function} axis - */ - function axis(selection) { - var - /** - * The orientation of the axis - * (see {@link axis#orient}) - * @param {string} [orient='bottom'] - * @memberof axis# - * @property - */ - orient = 'bottom', - // direction of the axis - - /** - * Amount of horizontal space (in pixels) avaible to render the axis in - * (see {@link axis#spaceX}) - * @param {number} [spaceX=0] - * @memberof axis# - * @property - */ - spaceX = 0, - - /** - * Amount of vertical space (in pixels) avaible to render the axis in - * (see {@link axis.spaceY}) - * @param {number} [spaceY=0] - * @memberof axis# - * @property - */ - spaceY = 0, - - - /** - * Whether or not to allow axis to render elements pass the main spatial dimension - * given the orientation (see {@link axis#orient}), where {@link axis#orient}="bottom" or {@link axis#orient}="top" - * the main dimension is {@link axis#spaceX} and where {@link axis#orient}="left" or {@link axis#orient}="right" - * the main dimension is {@link axis#spaceY} - * @param {boolean} [overflowQ=false] - * @memberof axis# - * @property - */ - overflowQ = false, - // whether or not to allow overflow - /** - * Whether or not the axis labels are for categorical data. If false, - * will use {@link axis#scale} to position ticks. - * @param {boolean} [categoricalQ=false] - * @memberof axis# - * @property - */ - categoricalQ = false, - // whether or not the axis is showing values or groups - /** - * Whether or not the axis ticks should have guidelines - * @param {boolean} [categoricalQ=false] - * @memberof axis# - * @property - */ - guideLinesQ = false, - // whether or not to allow overflow - - - /** - * How to group the tick labels - * @param {Array[]} [grouping=undefined] list of putatively other lists, which should correspond to tickLabels - * will space tick labels in nested lists closer together than outer lists - * @memberof axis# - * @property - */ - grouping, - - - /** - * The scale for which non-categorial (see {@link axis#categoricalQ}) ticks should be spaced - * @param {d3.scale} [scale=d3.scaleLinear] - * @memberof axis# - * @property - */ - - scale = d3.scaleLinear(), - - /** - * The padding for the domain of the scale (see {@link axis#scale}) - * @param {d3.scale} [scale=d3.scaleLinear] - * @memberof axis# - * @property - */ - domainPadding = 0.5, - - - /** - * Default space for the spacer (percentage) of main dimension given the orientation - * (see {@link axis#orient}), where {@link axis#orient}="bottom" or {@link axis#orient}="top" - * the main dimension is {@link axis#spaceX} and where {@link axis#orient}="left" or {@link axis#orient}="right" - * the main dimension is {@link axis#spaceY}between ticks - * @param {number} [objectSpacer=0.05] - * @memberof axis# - * @property - */ - objectSpacer = 0.05, - - /** - * The minimum size that an object can be if {@link axis#categoricalQ} is set to true - * @param {number} [minObjectSize=15] - * @memberof axis# - * @property - */ - minObjectSize = 15, - - /** - * The maximum size that an object can be if {@link axis#categoricalQ} is set to true - * @param {number} [maxObjectSize=15] - * @memberof axis# - * @property - */ - maxObjectSize = 50, - - - /** - * Color of the background - * @param {string} [backgroundFill="transparent"] - * @memberof axis# - * @property - */ - backgroundFill = 'transparent', - - /** - * Namespace for all items made by this instance of axis - * @param {string} [namespace="d3sm-axis"] - * @memberof axis# - * @property - */ - namespace = 'd3sm-axis', - - /** - * Class name for tick container ( element) - * @param {string} [objectClass="tick-group"] - * @memberof axis# - * @property - */ - objectClass = 'tick-group', - - - /** - * Values to show at each tick. Only used if categoricalQ is set true. See {@link axis#categoricalQ} - * @param {string[]} [tickLabels=undefined] - * @memberof axis# - * @property - */ - tickLabels, - // what to place at ticks - /** - * Values to show at each tick. Only used if categoricalQ is set false. See {@link axis#categoricalQ} - * @param {string[] | number[]} [objectClass=undefined] - * @memberof axis# - * @property - */ - tickValues, - // where to place ticks if not - /** - * Number of ticks to display if categoricalQ is false. See {@link axis#categoricalQ} - * @param {number} [numberOfTicks=5] - * @memberof axis# - * @property - */ - numberOfTicks = 5, - - - /** - * Stroke color of the main axis line - * @param {string} [lineStroke='black'] - * @memberof axis# - * @property - */ - lineStroke = 'black', - - /** - * Stroke width of the main axis line - * @param {number} [lineStrokeWidth=3] - * @memberof axis# - * @property - */ - lineStrokeWidth = 3, - - - /** - * Stroke color of ticks - * @param {string} [tickStroke='black'] - * @memberof axis# - * @property - */ - tickStroke = 'black', - - /** - * Stroke number of ticks - * @param {string} [tickStrokeWidth=2] - * @memberof axis# - * @property - */ - tickStrokeWidth = 2, - - /** - * Length - in pixels - of ticks - * @param {number} [tickLength=10] - * @memberof axis# - * @property - */ - tickLength = 10, - tickTickLabelSpacer = 10, - tickLabelMargin = 10, - - - /** - * Font size of tick labels - * @param {number} [tickLabelFontSize=14] - * @memberof axis# - * @property - */ - tickLabelFontSize = 14, - - /** - * Min font size of tick labels - * @param {number} [tickLabelMinFontSize=8] - * @memberof axis# - * @property - */ - tickLabelMinFontSize = 8, - - /** - * Max font size of tick labels - * @param {number} [tickLabelMaxFontSize=20] - * @memberof axis# - * @property - */ - tickLabelMaxFontSize = 20, - - - /** - * Text anchor of tick labels - * @param {string} [tickLabelTextAnchor="middle"] - * @memberof axis# - * @property - */ - tickLabelTextAnchor, - - /** - * Rotation of tick labels - * @param {number} [tickLabelRotation=0] - * @memberof axis# - * @property - */ - tickLabelRotation, - - /** - * Optional function for extracting the tick label from data - * @param {function} [tickLabelFunc=undefined] - * @memberof axis# - * @property - */ - tickLabelFunc = undefined, - - - /** - * Optional function for what to do when label is clicked - * @param {function} [tickLabelOnClick=function(d, i){}] - * @memberof axis# - * @property - */ - tickLabelOnClick = function tickLabelOnClick(d, i) {}, - - - /** - * Optional function for what to do when label is hovered - * @param {function} [tickLabelOnHoverFunc=function(d, i){}] - * @memberof axis# - * @property - */ - tickLabelOnHoverFunc = function tickLabelOnHoverFunc(d, i) { - return String(d).replace('-', ' ').replace('_', ' '); - }, - - - /** - * Length of guidelines - * @param {function} [guidelineSpace=undefined] - * @memberof axis# - * @property - */ - guidelineSpace, - - /** - * Stroke color of guidlines - * @param {string} [guidelineSpace="#333333"] - * @memberof axis# - * @property - */ - guideLineStroke = '#333333', - - /** - * Stroke width of guidlines - * @param {number} [guidelineSpace=2] - * @memberof axis# - * @property - */ - guideLineStrokeWidth = 2, - - - /** - * Duration of all transitions of this element - * @param {number} [transitionDuration=1000] - * @memberof axis# - * @property - */ - transitionDuration = 1000, - - /** - * Easing function for transitions - * @param {d3.ease} [easeFunc=d3.easeExp] - * @memberof axis# - * @property - */ - easeFunc = d3.easeExp, - - - /** - * Closure variable for getting object size after calculation - * @param {number} [objectSize=undefined] - * @memberof axis# - * @property - */ - objectSize, - - /** - * Closure variable for getting spacer size after calculation - * @param {number} [spacerSize=undefined] - * @memberof axis# - * @property - */ - spacerSize, - - - /** - * Decimal percision to round numerical tick labels to - * @param {number} [roundTo=2] - * @memberof axis# - * @property - */ - roundTo = 2, - label, - reverseScaleQ = false; - - axis.label = function (_) { - return arguments.length ? (label = _, axis) : label; - }; - axis.tickTickLabelSpacer = function (_) { - return arguments.length ? (tickTickLabelSpacer = _, axis) : tickTickLabelSpacer; - }; - axis.tickLabelMargin = function (_) { - return arguments.length ? (tickLabelMargin = _, axis) : tickLabelMargin; - }; - /** - * Gets or sets the selection in which items are manipulated - * @param {d3.selection} [_=none] - * @returns {axis | d3.selection} - * @memberof axis - * @property - * by default selection = selection - */ - - axis.selection = function (_) { - return arguments.length ? (selection = _, axis) : selection; - }; - - /** - * Gets or sets the orientation in which items are manipulated - * (see {@link axis#orient}) - * @param {string} [_=none] should be horizontal or vertical - * @returns {axis | string} - * @memberof axis - * @property - * by default orient="bottom" - */ - axis.orient = function (_) { - return arguments.length ? (orient = _, axis) : orient; - }; - /** - * Gets or sets the amount of horizontal space in which items are manipulated - * (see {@link axis#spaceX}) - * @param {number} [_=none] should be a number > 0 - * @returns {axis | number} - * @memberof axis - * @property - * by default spaceX = undefined - */ - axis.spaceX = function (_) { - return arguments.length ? (spaceX = _, axis) : spaceX; - }; - /** - * Gets or sets the amount of vertical space in which items are manipulated - * (see {@link axis#spaceY}) - * @param {number} [_=none] should be a number > 0 - * @returns {axis | number} - * @memberof axis - * @property - * by default spaceY = undefined - */ - axis.spaceY = function (_) { - return arguments.length ? (spaceY = _, axis) : spaceY; - }; - - /** - * Gets / sets whether or not axis is allowed to go beyond specified dimensions - * (see {@link axis#spaceX}) - * @param {boolean} [_=none] - * @returns {axis | boolean} - * @memberof axis - * @property - * by default overflowQ = false - */ - axis.overflowQ = function (_) { - return arguments.length ? (overflowQ = _, axis) : overflowQ; - }; - /** - * Gets / sets whether or not axis will display categorial ticks or by numerical value - * (see {@link axis#categoricalQ}) - * @param {boolean} [_=none] - * @returns {axis | boolean} - * @memberof axis - * @property - * by default categoricalQ = false - */ - axis.categoricalQ = function (_) { - return arguments.length ? (categoricalQ = _, axis) : categoricalQ; - }; - /** - * Gets / sets whether or not axis ticks should have guidelines - * (see {@link axis#guideLinesQ}) - * @param {boolean} [_=none] - * @returns {axis | boolean} - * @memberof axis - * @property - * by default guideLinesQ = false - */ - axis.guideLinesQ = function (_) { - return arguments.length ? (guideLinesQ = _, axis) : guideLinesQ; - }; - - /** - * Gets / sets how ticks should be groupped - * (see {@link axis#grouping}) - * @param {Array[]} [_=none] list of putatively other lists, which should correspond to tickLabels - * will space tick labels in nested lists closer together than outer lists - * @returns {axis | Array[]} - * @memberof axis - * @property - * by default grouping = undefined - */ - axis.grouping = function (_) { - return arguments.length ? (grouping = _, axis) : grouping; - }; - - /** - * Gets / sets the scale for which non-categorial ticks should - * be spaced - * (see {@link axis#scale}) - * @param {d3.scale} [_=none] - * @returns {axis | d3.scale} - * @memberof axis - * @property - * by default scale = d3.scaleLinear() - */ - axis.scale = function (_) { - return arguments.length ? (scale = _, axis) : scale; - }; - /** - * Gets / sets the padding for the domain of the scale - * (see {@link axis#domainPadding}) - * @param {number} [_=none] - * @returns {axis | number} - * @memberof axis - * @property - * by default domainPadding = 0.5 - */ - axis.domainPadding = function (_) { - return arguments.length ? (domainPadding = _, axis) : domainPadding; - }; - - /** - * Gets / sets objectSpacer - * (see {@link axis#objectSpacer}) - * @param {number} [_=none] - * @returns {axis | number} - * @memberof axis - * @property - * by default objectSpacer = 0.05 - */ - axis.objectSpacer = function (_) { - return arguments.length ? (objectSpacer = _, axis) : objectSpacer; - }; - /** - * Gets / sets the minObjectSize - * (see {@link axis#minObjectSize}) - * @param {number} [_=none] - * @returns {axis | number} - * @memberof axis - * @property - * by default minObjectSize = 15 - */ - axis.minObjectSize = function (_) { - return arguments.length ? (minObjectSize = _, axis) : minObjectSize; - }; - /** - * Gets / sets the maxObjectSize - * (see {@link axis#maxObjectSize}) - * @param {number} [_=none] - * @returns {axis | number} - * @memberof axis - * @property - * by default maxObjectSize = 50 - */ - axis.maxObjectSize = function (_) { - return arguments.length ? (maxObjectSize = _, axis) : maxObjectSize; - }; - - /** - * Gets / sets the namespace - * (see {@link axis#namespace}) - * @param {string} [_=none] - * @returns {axis | string} - * @memberof axis - * @property - * by default namespace = 'd3sm-axis' - */ - axis.namespace = function (_) { - return arguments.length ? (namespace = _, axis) : namespace; - }; - /** - * Gets / sets the backgroundFill - * (see {@link axis#backgroundFill}) - * @param {string} [_=none] - * @returns {axis | string} - * @memberof axis - * @property - * by default backgroundFill = 'transparent' - */ - axis.backgroundFill = function (_) { - return arguments.length ? (backgroundFill = _, axis) : backgroundFill; - }; - /** - * Gets / sets the objectClass - * (see {@link axis#objectClass}) - * @param {string} [_=none] - * @returns {axis | string} - * @memberof axis - * @property - * by default objectClass = 'tick-group' - */ - axis.objectClass = function (_) { - return arguments.length ? (objectClass = _, axis) : objectClass; - }; - - /** - * Gets / sets the tickLabels - * (see {@link axis#tickLabels}) - * @param {string[]} [_=none] - * @returns {axis | string[]} - * @memberof axis - * @property - * by default tickLabels = undefined - */ - axis.tickLabels = function (_) { - return arguments.length ? (tickLabels = _, axis) : tickLabels; - }; - /** - * Gets / sets the tickValues - * (see {@link axis#tickValues}) - * @param {number[]} [_=none] - * @returns {axis | number[]} - * @memberof axis - * @property - * by default tickValues = undefined - */ - axis.tickValues = function (_) { - return arguments.length ? (tickValues = _, axis) : tickValues; - }; - /** - * Gets / sets the tickValues - * (see {@link axis#numberOfTicks}) - * @param {number} [_=none] - * @returns {axis | number} - * @memberof axis - * @property - * by default numberOfTicks = 5 - */ - axis.numberOfTicks = function (_) { - return arguments.length ? (numberOfTicks = _, axis) : numberOfTicks; - }; - - /** - * Gets / sets the lineStroke - * (see {@link axis#lineStroke}) - * @param {string} [_=none] - * @returns {axis | string} - * @memberof axis - * @property - * by default lineStroke = 'black' - */ - axis.lineStroke = function (_) { - return arguments.length ? (lineStroke = _, axis) : lineStroke; - }; - /** - * Gets / sets the lineStrokeWidth - * (see {@link axis#lineStrokeWidth}) - * @param {number} [_=none] - * @returns {axis | number} - * @memberof axis - * @property - * by default lineStrokeWidth = 3 - */ - axis.lineStrokeWidth = function (_) { - return arguments.length ? (lineStrokeWidth = _, axis) : lineStrokeWidth; - }; - - /** - * Gets / sets the tickStroke - * (see {@link axis#tickStroke}) - * @param {string} [_=none] - * @returns {axis | string} - * @memberof axis - * @property - * by default tickStroke = 'black' - */ - axis.tickStroke = function (_) { - return arguments.length ? (tickStroke = _, axis) : tickStroke; - }; - /** - * Gets / sets the tickStrokeWidth - * (see {@link axis#tickStrokeWidth}) - * @param {number} [_=none] - * @returns {axis | number} - * @memberof axis - * @property - * by default tickStrokeWidth = 2 - */ - axis.tickStrokeWidth = function (_) { - return arguments.length ? (tickStrokeWidth = _, axis) : tickStrokeWidth; - }; - /** - * Gets / sets the tickLength - * (see {@link axis#tickLength}) - * @param {number} [_=none] - * @returns {axis | number} - * @memberof axis - * @property - * by default tickLength = 10 - */ - axis.tickLength = function (_) { - return arguments.length ? (tickLength = _, axis) : tickLength; - }; - - /** - * Gets / sets the tickLabelFontSize - * (see {@link axis#tickLabelFontSize}) - * @param {number} [_=none] - * @returns {axis | number} - * @memberof axis - * @property - * by default tickLabelFontSize = 14 - */ - axis.tickLabelFontSize = function (_) { - return arguments.length ? (tickLabelFontSize = _, axis) : tickLabelFontSize; - }; - /** - * Gets / sets the tickLabelMinFontSize - * (see {@link axis#tickLabelMinFontSize}) - * @param {number} [_=none] - * @returns {axis | number} - * @memberof axis - * @property - * by default tickLabelMinFontSize = 8 - */ - axis.tickLabelMinFontSize = function (_) { - return arguments.length ? (tickLabelMinFontSize = _, axis) : tickLabelMinFontSize; - }; - /** - * Gets / sets the tickLabelMaxFontSize - * (see {@link axis#tickLabelMaxFontSize}) - * @param {number} [_=none] - * @returns {axis | number} - * @memberof axis - * @property - * by default tickLabelMaxFontSize = 20 - */ - axis.tickLabelMaxFontSize = function (_) { - return arguments.length ? (tickLabelMaxFontSize = _, axis) : tickLabelMaxFontSize; - }; - - /** - * Gets / sets the tickLabelTextAnchor - * (see {@link axis#tickLabelTextAnchor}) - * @param {string} [_=none] - * @returns {axis | string} - * @memberof axis - * @property - * by default tickLabelTextAnchor = 'center' - */ - axis.tickLabelTextAnchor = function (_) { - return arguments.length ? (tickLabelTextAnchor = _, axis) : tickLabelTextAnchor; - }; - /** - * Gets / sets the tickLabelRotation - * (see {@link axis#tickLabelRotation}) - * @param {number} [_=none] - * @returns {axis | number} - * @memberof axis - * @property - * by default tickLabelRotation = 0 - */ - axis.tickLabelRotation = function (_) { - return arguments.length ? (tickLabelRotation = _, axis) : tickLabelRotation; - }; - /** - * Gets / sets the tickLabelFunc - * (see {@link axis#tickLabelFunc}) - * @param {function} [_=none] - * @returns {axis | function} - * @memberof axis - * @property - * by default tickLabelFunc = undefined - */ - axis.tickLabelFunc = function (_) { - return arguments.length ? (tickLabelFunc = _, axis) : tickLabelFunc; - }; - - /** - * Gets / sets the tickLabelOnClick - * (see {@link axis#tickLabelOnClick}) - * @param {function} [_=none] - * @returns {axis | function} - * @memberof axis - * @property - */ - axis.tickLabelOnClick = function (_) { - return arguments.length ? (tickLabelOnClick = _, axis) : tickLabelOnClick; - }; - - /** - * Gets / sets the guidelineSpace - * (see {@link axis#guidelineSpace}) - * @param {number} [_=none] - * @returns {axis | number} - * @memberof axis - * @property - * by default guidelineSpace = undefined - */ - axis.guidelineSpace = function (_) { - return arguments.length ? (guidelineSpace = _, axis) : guidelineSpace; - }; - /** - * Gets / sets the guideLineStroke - * (see {@link axis#guideLineStroke}) - * @param {string} [_=none] - * @returns {axis | string} - * @memberof axis - * @property - * by default guideLineStroke = "#333333" - */ - axis.guideLineStroke = function (_) { - return arguments.length ? (guideLineStroke = _, axis) : guideLineStroke; - }; - /** - * Gets / sets the guideLineStrokeWidth - * (see {@link axis#guideLineStrokeWidth}) - * @param {number} [_=none] - * @returns {axis | number} - * @memberof axis - * @property - * by default guideLineStrokeWidth = 2 - */ - axis.guideLineStrokeWidth = function (_) { - return arguments.length ? (guideLineStrokeWidth = _, axis) : guideLineStrokeWidth; - }; - - /** - * Gets / sets the transitionDuration - * (see {@link axis#transitionDuration}) - * @param {number} [_=none] - * @returns {axis | number} - * @memberof axis - * @property - * by default transitionDuration = 1000 - */ - axis.transitionDuration = function (_) { - return arguments.length ? (transitionDuration = _, axis) : transitionDuration; - }; - /** - * Gets / sets the easeFunc - * (see {@link axis#easeFunc}) - * @param {d3.ease} [_=none] - * @returns {axis | d3.ease} - * @memberof axis - * @property - * by default easeFunc = d3.easeExp - */ - axis.easeFunc = function (_) { - return arguments.length ? (easeFunc = _, axis) : easeFunc; - }; - - /** - * Gets / sets the objectSize - * (see {@link axis#objectSize}) - * @param {number} [_=none] - * @returns {axis | number} - * @memberof axis - * @property - * by default objectSize = undefined - */ - axis.objectSize = function (_) { - return arguments.length ? (objectSize = _, axis) : objectSize; - }; - /** - * Gets / sets the spacerSize - * (see {@link axis#spacerSize}) - * @param {number} [_=none] - * @returns {axis | number} - * @memberof axis - * @property - * by default spacerSize = undefined - */ - axis.spacerSize = function (_) { - return arguments.length ? (spacerSize = _, axis) : spacerSize; - }; - - /** - * Gets / sets the roundTo - * (see {@link axis#roundTo}) - * @param {number} [_=none] - * @returns {axis | number} - * @memberof axis - * @property - * by default roundTo = 2 - */ - axis.roundTo = function (_) { - return arguments.length ? (roundTo = _, axis) : roundTo; - }; - axis.reverseScaleQ = function (_) { - return arguments.length ? (reverseScaleQ = _, axis) : reverseScaleQ; - }; - - axis.tickLabelOnHoverFunc = function (_) { - return arguments.length ? (tickLabelOnHoverFunc = _, axis) : tickLabelOnHoverFunc; - }; - - function axis() { - // for convenience in handling orientation specific values - var horizontalQ = hasQ(['top', 'bottom', 'horizontal'], orient) ? true : false; - var verticalQ = !horizontalQ; - - // background cliping rectangle - var bgcpRect = { x: 0, y: 0, width: spaceX, height: spaceY - // modify the rect based on axis orientation - };if (orient == "left") { - bgcpRect.x -= spaceX; - if (guideLinesQ) { - bgcpRect.width += guidelineSpace; - } /* these two lines increase the clipping rect to allow for text at the edge of the axis */ - bgcpRect.y -= tickLabelMaxFontSize; - bgcpRect.height += 2 * tickLabelMaxFontSize; - } - if (orient == "bottom") { - bgcpRect.y = bgcpRect.y; - if (guideLinesQ) { - bgcpRect.y -= guidelineSpace;bgcpRect.height += guidelineSpace; - } /* these two lines increase the clipping rect to allow for text at the edge of the axis */ - bgcpRect.x -= tickLabelMaxFontSize; - bgcpRect.width += 2 * tickLabelMaxFontSize; - } - if (orient == "top") { - bgcpRect.y -= spaceY; - if (guideLinesQ) { - bgcpRect.height += guidelineSpace; - } /* these two lines increase the clipping rect to allow for text at the edge of the axis */ - bgcpRect.y -= tickLabelMaxFontSize; - bgcpRect.height += 2 * tickLabelMaxFontSize; - } - if (orient == "right") { - bgcpRect.x = 0; - if (guideLinesQ) { - bgcpRect.width += guidelineSpace;bgcpRect.x -= guidelineSpace; - } /* these two lines increase the clipping rect to allow for text at the edge of the axis */ - bgcpRect.y -= tickLabelMaxFontSize; - bgcpRect.height += 2 * tickLabelMaxFontSize; - } - - var container = setupContainer(selection, namespace, bgcpRect, backgroundFill); - - // defaults for text-anchor and text rotation - if (orient == 'top') { - tickLabelTextAnchor = tickLabelTextAnchor == undefined ? 'start' : tickLabelTextAnchor; - tickLabelRotation = tickLabelRotation == undefined ? -90 : tickLabelRotation; - } - if (orient == 'bottom') { - tickLabelTextAnchor = tickLabelTextAnchor == undefined ? 'end' : tickLabelTextAnchor; - tickLabelRotation = tickLabelRotation == undefined ? -90 : tickLabelRotation; - } - if (orient == 'left') { - tickLabelTextAnchor = tickLabelTextAnchor == undefined ? 'end' : tickLabelTextAnchor; - tickLabelRotation = tickLabelRotation == undefined ? 0 : tickLabelRotation; - } - if (orient == 'right') { - tickLabelTextAnchor = tickLabelTextAnchor == undefined ? 'start' : tickLabelTextAnchor; - tickLabelRotation = tickLabelRotation == undefined ? 0 : tickLabelRotation; - } - - /* - If categorical: - -> use grouping if defined, - -> else use the labels provided - else: - if grouping undefined - and no specified number of tickes - -> make numberOfTick ticks - -> else use provided tick values - -> use grouping - */ - var tickData = categoricalQ ? grouping == undefined ? tickLabels : grouping : grouping == undefined ? numberOfTicks != undefined ? - // ? (tickValues.length < numberOfTicks) - tickRange.apply(undefined, toConsumableArray(d3.extent(tickValues)).concat([numberOfTicks])) : tickValues : grouping; - - var flatTickData = flatten(tickData); - var numberOfObjects = flatTickData.length; - var space = horizontalQ ? spaceX : spaceY; - var extent = d3.extent(flatTickData); - - if (reverseScaleQ) { - extent.reverse(); - } - var domain = reverseScaleQ ? [extent[0] + domainPadding, extent[1] - domainPadding] : [extent[0] - domainPadding, extent[1] + domainPadding]; - - scale.domain(domain).range([horizontalQ ? 0 : spaceY, horizontalQ ? spaceX : 0]); - - /* - Scales are based on the values of the chart and correspond to the spacings of the - chart. If the chart has already been rendered, these values (expensive to caluclate) can - be passed to axis to prevent recalculation. - */ - - // calculate object size if needed - objectSize = objectSize == undefined ? calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ) : objectSize; - - // calculate spacer size if needed - spacerSize = spacerSize == undefined ? calculateWidthOfSpacer(flatTickData, space, objectSize, numberOfObjects, objectSpacer, overflowQ) : spacerSize; - - var objClass = hypenate(namespace, categoricalQ ? objectClass + '-categorical' : objectClass); - - var spacerFunction = groupingSpacer().horizontalQ(horizontalQ).scale(scale).moveby(categoricalQ ? 'category' : 'scale').numberOfObjects(numberOfObjects).objectClass(objClass).objectSize(objectSize).spacerSize(spacerSize).transitionDuration(transitionDuration).easeFunc(easeFunc).namespace(namespace); - - var tickEnterAnimation = function tickEnterAnimation(sel) { - var mt = scale(sel.datum()), - dist = scale(extent[1]) * 2, - k = mt < extent[1] / 2 ? 1 : -1; - k = horizontalQ ? k * -1 : k; - sel.attr('transform', function (d, i) { - var x = horizontalQ ? dist * k : 0, - y = !horizontalQ ? dist * k : 0, - t = 'translate(' + x + ',' + y + ')'; - return t; - }); - }; - var tickExitAnimation = function tickExitAnimation(sel) { - var mt = scale(sel.datum()), - dist = scale(extent[1]) * 2, - k = mt < extent[1] / 2 ? 1 : -1; - k = horizontalQ ? k * -1 : k; - sel.transition().duration(transitionDuration).ease(easeFunc).style('opacity', 0).attr('transform', function (d, i) { - var x = horizontalQ ? dist * k : 0, - y = !horizontalQ ? dist * k : 0, - t = 'translate(' + x + ',' + y + ')'; - return t; - }).remove(); - }; - - if (!categoricalQ) { - spacerFunction.enterFunction(tickEnterAnimation); - spacerFunction.exitFunction(tickExitAnimation); - } - - // move tick containers - spacerFunction(container, tickData, 0); - - // move by for x and y needed to center categorical ticks, labels, and guidelines - function moveXBy(d, i, horizontalQ, categoricalQ, objectSize) { - return horizontalQ ? categoricalQ ? objectSize / 2 : 0 : 0; - } - - function moveYBy(d, i, verticalQ, categoricalQ, objectSize) { - return verticalQ ? categoricalQ ? objectSize / 2 : 0 : 0; - } - - var labelNameGroup = safeSelect(selection, 'g', hypenate(namespace, 'axis-name')); - - var labelElement = safeSelect(labelNameGroup, 'text', hypenate(namespace, 'name')); - if (labelElement != undefined) { - labelElement.text(label); - - if (orient == 'left' || orient == 'right') { - labelElement.attr('transform', 'rotate(-90)'); - } - - var bbox = labelElement.node().getBoundingClientRect(); - labelNameGroup.attr('transform', function (d, i) { - var x = 0, - y = 0, - t; - - if (orient == 'bottom') { - x = spaceX - bbox.width - tickLabelMargin; - y = -tickTickLabelSpacer; - } else if (orient == 'top') { - x = spaceX - bbox.width - tickLabelMargin; - x = tickLabelMargin; - y = bbox.height + tickTickLabelSpacer; - } else if (orient == 'left') { - x = bbox.width + tickTickLabelSpacer; - y = bbox.height + tickLabelMargin; - } else if (orient == 'right') { - x = -(bbox.width + tickTickLabelSpacer); - y = bbox.height + tickLabelMargin; - } - t = 'translate(' + x + ',' + y + ')'; - return t; - }); - } else { - labelElement.remove(); - } - /* - Idea from Stack Overflow - https://stackoverflow.com/questions/50579535/d3-js-v4-truncate-text-to-fit-in-fixed-space/50585022?noredirect=1#comment88235562_50585022 - to use clip path to make things fit in fixed size. Have yet got this to work nicely. - */ - // var defs = d3.select(container.node().parentNode).select('defs') - // var tickLabelClipPath = safeSelect(defs, 'clipPath', hypenate(namespace,'tick-label-clip-path')).attr('id', hypenate(namespace,'tick-label-clip-path')) - // var tickLabelClipPathRect = safeSelect(tickLabelClipPath, 'rect', hypenate(namespace,'tick-label-clip-path-rect')) - // .attr('x', 0) - // .attr('y', 0) - // .attr('width', function(d, i){ - // if (horizontalQ) { return tickLabelFontSize } - // if (verticalQ) { return spaceX - tickLength } - // }) - // .attr('height', function(d, i){ - // if (verticalQ) { return tickLabelFontSize } - // if (horizontalQ) { return spaceY - tickLength } - // }) - - - // for each tick container - var ticks = container.selectAll('g:not(.to-remove).' + objClass).each(function (d, i) { - var that = d3.select(this).style('opacity', 1); - - // make and move tick - var tick = safeSelect(that, 'line', hypenate(namespace, 'tick')).attr("x1", 0).attr("x2", horizontalQ ? 0 : orient == "left" ? -tickLength : tickLength).attr("y1", 0).attr('y2', verticalQ ? 0 : orient == "top" ? -tickLength : tickLength).attr('stroke', tickStroke).attr('stroke-width', tickStrokeWidth).attr('transform', function (d, i) { - var x = moveXBy(d, i, horizontalQ, categoricalQ, objectSize), - y = moveYBy(d, i, verticalQ, categoricalQ, objectSize), - t = 'translate(' + x + ',' + y + ')'; - return t; - }); - - // make and move label - var label = safeSelect(that, 'text', hypenate(namespace, 'label')).text(function (d, i) { - var s = typeof d == 'number' ? round(d, roundTo) : d; - s = truncateString(String(s), (horizontalQ ? spaceY : spaceX) - tickLength - tickLabelMargin - tickTickLabelSpacer, tickLabelFontSize * 0.45); - return s; - }).attr('font-size', tickLabelFontSize).attr('text-anchor', tickLabelTextAnchor); - // truncateText(label, label.text(), orient, tickLength, horizontalQ ? spaceY : spaceX, overflowQ) - - label.attr('transform', function (d, i) { - var rect = d3.select(this).node().getBoundingClientRect(), - leng = d3.select(this).node().getComputedTextLength(), - x = moveXBy(d, i, horizontalQ, categoricalQ, objectSize), - y = moveYBy(d, i, verticalQ, categoricalQ, objectSize); - - if (orient == 'top') { - y = -(tickLength + tickTickLabelSpacer); - // y = tickLength+tickTickLabelSpacer; - - // y -= Math.max(rect.height, rect.width); - x += Math.min(rect.height, rect.width) * 0.25; - // x -= leng * 0.25 + s - } - if (orient == 'bottom') { - y = tickLength + tickTickLabelSpacer; - x += Math.min(rect.height, rect.width) * 0.25; - // x += leng * 0.25 - s - } - if (orient == 'left') { - x -= tickLength + tickTickLabelSpacer; - // y += rect.height * 0.5; y-= rect.height/4 - y += Math.min(rect.height, rect.width) * 0.25; - // y += leng * 0.25 - } - if (orient == 'right') { - x += tickLength + tickTickLabelSpacer; - // y += Math.min(rect.height, rect.width) * 0.25 - // y += leng * 0.25 - y += rect.height * 0.5;y -= rect.height / 4; - } - - var t = 'translate(' + x + ',' + y + ')', - r = 'rotate(' + tickLabelRotation + ')'; - return t + r; - }).on('mousemove', labelHover).on('mouseout', labelHoverOff).on('click', tickLabelOnClick); - // .attr('clip-path', 'url(#'+hypenate(namespace,'tick-label-clip-path')+')') - - // add guidlines as needed - if (guideLinesQ) { - var gline = safeSelect(that, 'line', hypenate(namespace, 'guideline')).transition().duration(transitionDuration).ease(easeFunc).attr("x1", 0).attr("x2", horizontalQ ? 0 : orient == "left" ? guidelineSpace : -guidelineSpace).attr("y1", 0).attr('y2', verticalQ ? 0 : orient == "top" ? guidelineSpace : -guidelineSpace).attr('transform', function (d, i) { - var x = moveXBy(d, i, horizontalQ, categoricalQ, objectSize), - y = moveYBy(d, i, verticalQ, categoricalQ, objectSize), - t = 'translate(' + x + ',' + y + ')'; - return t; - }); - } else { - that.select('line.' + hypenate(namespace, 'guideline')).remove(); - } - }); - - // apply alternating guidline thickness - if (guideLinesQ) { - container.selectAll('.' + hypenate(namespace, 'guideline')).attr('stroke', function (d, i) { - if (i % 2 == 0) { - return modifyHexidecimalColorLuminance(guideLineStroke, 0.8); - } - return guideLineStroke; - }).attr('stroke-width', function (d, i) { - if (i % 2 == 0) { - return guideLineStrokeWidth * 0.8; - } - return guideLineStrokeWidth; - }).attr('minor', function (d, i) { - return i % 2 == 0; - }); - } - - /*************************************************************************** - ** Make the line of the axis - ***************************************************************************/ - var line = safeSelect(selection, 'path', hypenate(namespace, 'line')) - // .attr('x1', 0) - // .attr('x2', horizontalQ ? spaceX : 0) - // .attr('y1', 0) - // .attr('y2', horizontalQ ? 0 : spaceY) - .attr('d', horizontalQ ? 'M 0,0 H' + spaceX + ',0' : 'M 0,0 V 0,' + spaceY).attr('stroke', lineStroke).attr('stroke-width', lineStrokeWidth).classed('axis-line', true); - } - - // hover of label show full text label in case it is truncated - function labelHover(d, i) { - var t = d3.select(this).style('fill', 'red'); - d3.select(t.node().parentNode).select("line." + hypenate(namespace, 'tick')).attr("stroke", 'red').attr("stroke-width", tickStrokeWidth * 2); - - if (guideLinesQ) { - d3.select(t.node().parentNode).select('line.' + hypenate(namespace, 'guideline')).attr('stroke', 'red').attr('stroke-width', guideLineStrokeWidth * 2); - } - - var s = typeof d == 'number' ? round(d, roundTo) : d; - - var m = d3.mouse(d3.select('html').node()); - var div = safeSelect(d3.select('body'), 'div', hypenate(namespace, 'guideline-tooltip')).attr('id', hypenate(namespace, 'guideline-tooltip')).style('position', 'absolute').style('left', d3.event.pageX + 15 + 'px').style('top', d3.event.pageY + 15 + 'px').style('background-color', 'white').style('border-color', 'black') - // .style('min-width', (tickLabelFontSize * (String(s).split('.')[0].length+3))+'px') - // .style('min-height', (tickLabelFontSize * (String(s).split('.')[0].length+3))+'px') - .style('border-radius', '10px').style('display', 'flex').style('justify-content', 'center').style('text-align', 'middle').style('padding', 4 + "px").style('border-style', 'solid').style('border-width', 2); - - var text = safeSelect(div, 'div').text(tickLabelOnHoverFunc(s, i)).style('color', 'black').style('align-self', 'center'); - - var bbox = div.node().getBoundingClientRect(); - if (bbox.x + bbox.width > window.innerWidth) { - div.style('left', d3.event.pageX - 15 - 300 + 'px'); - } - } - - function labelHoverOff(d, i) { - var t = d3.select(this).style('fill', 'black'); - d3.select(t.node().parentNode).select("line." + hypenate(namespace, 'tick')).attr("stroke", tickStroke).attr("stroke-width", tickStrokeWidth); - - if (guideLinesQ) { - var gline = d3.select(t.node().parentNode).select('line.' + hypenate(namespace, 'guideline')); - var minorQ = gline.attr('minor'); - gline.attr('stroke', function (d, ii) { - if (minorQ == 'true') { - return modifyHexidecimalColorLuminance(guideLineStroke, 0.8); - } - return guideLineStroke; - }).attr('stroke-width', function (d, ii) { - if (minorQ == 'true') { - return guideLineStrokeWidth * 0.8; - } - return guideLineStrokeWidth; - }); - } - d3.select("#" + hypenate(namespace, 'guideline-tooltip')).remove(); - } - - return axis; - } - - /** - * Creates a colorFunction - * @constructor colorFunction - * @namespace colorFunction - * @returns {function} colorFunction - */ - function colorFunction() { - var /** - * Default colors to use - * @param {number[]} [colors=["#2c7bb6", "#00a6ca", "#00ccbc", "#90eb9d", "#ffff8c", "#f9d057", "#f29e2e", "#e76818", "#d7191c"]] - * @memberof colorFunction# - * @property - */ - colors = ["#2c7bb6", "#00a6ca", "#00ccbc", "#90eb9d", "#ffff8c", "#f9d057", "#f29e2e", "#e76818", "#d7191c"], - - /** - * Interpolator for colors - * @param {d3.interpolation} [interpolation=d3.interpolateRgb] - * @memberof colorFunction# - * @property - */ - interpolation = d3.interpolateRgb, - - /** - * Function for modifying color luminance - * @param {function} [modifyOpacity=modifyHexidecimalColorLuminance] - * @memberof colorFunction# - * @property - */ - modifyOpacity = modifyHexidecimalColorLuminance, - - /** - * How to modify color for stroke - * @param {number} [strokeOpacity=0] - * @memberof colorFunction# - * @property - */ - strokeOpacity = 0, - - /** - * How to modify color for fill - * @param {number} [fillOpacity=0.4] - * @memberof colorFunction# - * @property - */ - fillOpacity = 0.4, - - /** - * How to determine the color to use - * @param {string} [colorBy='index'] - * @memberof colorFunction# - * @property - */ - colorBy = 'index', - - /** - * Sets the scale for interpolating the colors - * @param {number[]} [dataExtent=[0, colors.length - 1]] - * @memberof colorFunction# - * @property - */ - dataExtent = [0, colors.length - 1], - - /** - * Extracts the value to color by - * @param {function} [valueExtractor=function(k, v, i) {return v}] - * @memberof colorFunction# - * @property - */ - valueExtractor = function valueExtractor(k, v, i) { - return v; - }, - - - /** - * Extracts the category to color by - * @param {function} [categoryExtractor=function(k, v, i) {return v.category}] - * @memberof colorFunction# - * @property - */ - categoryExtractor = function categoryExtractor(k, v, i) { - return v.category; - }, - - - /** - * The different type of categories of which to color by - * @param {string[]} [categories=undefined] - * @memberof colorFunction# - * @property - */ - categories, - - - /** - * Scale for interpolating the colors - * @param {d3.scale} [scale=d3.scaleLinear()] - * @memberof colorFunction# - * @property - */ - scale = d3.scaleLinear().interpolate(interpolation).domain(dataExtent).range(colors), - helperScale = d3.scaleLinear(); - - // var h = x => '#' + x.match(/\d+/g).map(y = z => ((+z < 16)?'0':'') + (+z).toString(16)).join(''); - var h = function h(x) { - return "#" + x.match(/\d+/g).map(function (y, i) { - return (+y < 16 ? '0' : '') + (+y).toString(16); - }).join(''); - }; - - /** - * Gets or sets the default colors - * (see {@link colorFunction#colors}) - * @param {number[]} [_=none] - * @returns {colorFunction | number[]} - * @memberof colorFunction - * @property - */ - colorFunction.colors = function (_) { - return arguments.length ? (colors = _, scale.range(colors), colorFunction) : colors; - }; - /** - * Gets or sets the function for interpolating the colors - * (see {@link colorFunction#interpolation}) - * @param {d3.interpolation} [_=none] - * @returns {colorFunction | d3.interpolation} - * @memberof colorFunction - * @property - */ - colorFunction.interpolation = function (_) { - return arguments.length ? (interpolation = _, scale.interpolate(interpolation).range(colors), colorFunction) : interpolation; - }; - /** - * Gets or sets the values for the scale which transforms the value to a color - * (see {@link colorFunction#dataExtent}) - * @param {number[]} [_=none] - * @returns {colorFunction | number[]} - * @memberof colorFunction - * @property - */ - colorFunction.dataExtent = function (_) { - return arguments.length ? (dataExtent = _, scale.domain(dataExtent).interpolate(scale.interpolate()), colorFunction) : dataExtent; - }; - /** - * Gets or sets the vthe scale which transforms the value to a color - * (see {@link colorFunction#scale}) - * @param {d3.scale} [_=none] - * @returns {colorFunction | d3.scale} - * @memberof colorFunction - * @property - */ - colorFunction.scale = function (_) { - return arguments.length ? (_ = _.domain(scale.domain()).interpolate(scale.interpolate()).range(scale.range()), scale = _, colorFunction) : scale; - }; - /** - * Gets or sets the function for modify opacity - * (see {@link colorFunction#modifyOpacity}) - * @param {function} [_=none] - * @returns {colorFunction | function} - * @memberof colorFunction - * @property - */ - colorFunction.modifyOpacity = function (_) { - return arguments.length ? (modifyOpacity = _, colorFunction) : modifyOpacity; - }; - /** - * Gets or sets the value to modify the color for the stroke via {@link colorFunction#modifyOpacity} - * (see {@link colorFunction#strokeOpacity}) - * @param {number} [_=none] - * @returns {colorFunction | number} - * @memberof colorFunction - * @property - */ - colorFunction.strokeOpacity = function (_) { - return arguments.length ? (strokeOpacity = _, colorFunction) : strokeOpacity; - }; - /** - * Gets or sets the value to modify the color for the stroke via {@link colorFunction#fillOpacity} - * (see {@link colorFunction#fillOpacity}) - * @param {number} [_=none] - * @returns {colorFunction | number} - * @memberof colorFunction - * @property - */ - colorFunction.fillOpacity = function (_) { - return arguments.length ? (fillOpacity = _, colorFunction) : fillOpacity; - }; - /** - * Gets or sets the value to colorBy - * (see {@link colorFunction#colorBy}) - * @param {string} [_=none] - * @returns {colorFunction | string} - * @memberof colorFunction - * @property - */ - colorFunction.colorBy = function (_) { - return arguments.length ? (colorBy = _, colorFunction) : colorBy; - }; - /** - * Gets or sets the value of valueExtractor - * (see {@link colorFunction#valueExtractor}) - * @param {function} [_=none] - * @returns {colorFunction | function} - * @memberof colorFunction - * @property - */ - colorFunction.valueExtractor = function (_) { - return arguments.length ? (valueExtractor = _, colorFunction) : valueExtractor; - }; - - /** - * Gets or sets the value of categoryExtractor - * (see {@link colorFunction#categoryExtractor}) - * @param {function} [_=none] - * @returns {colorFunction | function} - * @memberof colorFunction - * @property - */ - colorFunction.categoryExtractor = function (_) { - return arguments.length ? (categoryExtractor = _, colorFunction) : categoryExtractor; - }; - /** - * Gets or sets the value of categoryExtractor - * (see {@link colorFunction#categories}) - * @param {string[]} [_=none] - * @returns {colorFunction | string[]} - * @memberof colorFunction - * @property - */ - colorFunction.categories = function (_) { - return arguments.length ? (categories = _, colorFunction) : categories; - }; - - function colorFunction(key, value, index, type, hoverQ) { - var c, - opac = type == "fill" ? fillOpacity : strokeOpacity; - - updateScale(); - - if (colorBy == "index") { - c = type != undefined ? modifyOpacity(h(scale(index)), opac) : h(scale(index)); - } else if (colorBy == 'value') { - var v = valueExtractor(key, value, index); - // if (v < dataExtent[0]) {dataExtent[0] = v; updateScale()} - // if (v > dataExtent[1]) {dataExtent[1] = v; updateScale()} - - c = type != undefined ? modifyOpacity(h(scale(v)), opac) : h(scale(v)); - } else if (colorBy == 'category') { - var cat = categoryExtractor(key, value, index); - var v = categories.indexOf(cat); - c = type != undefined ? modifyOpacity(h(scale(v)), opac) : h(scale(v)); - } else { - c = type != undefined ? modifyOpacity(h(scale(index)), opac) : h(scale(index)); - } - - return c; - } - - function updateScale() { - - helperScale.domain([0, colors.length]); - if (colorBy == 'category' && categories != undefined) { - helperScale.range([0, categories.length]); - } else { - helperScale.range(dataExtent); - } - - var a = Array(colors.length).fill(0).map(function (d, i) { - return helperScale(i); - }); - scale.domain(a); - } - - return colorFunction; - } - - /******************************************************************************* - ** ** - ** ** - ** TOOLTIP ** - ** ** - ** ** - *******************************************************************************/ - /** - * Produces a function for handling the tooltip - * - * {@link https://sumneuron.gitlab.io/d3sm/demos/tooltip-design/index.html Demo} - * @param {d3.selection} selection - * @returns {tooltip} - * @namespace tooltip - */ - function tooltip(selection) { - - var keys, values, header, data, selection; - - /** - * Gets / sets the keys to be displayed in the tooltip. - * If not set, uses d3.keys(data[key]) - * @param {string[]} [_=none] - * @returns {tooltip | string[]} - * @memberof tooltip - */ - tooltip.keys = function (_) { - return arguments.length ? (keys = _, tooltip) : keys; - }; - /** - * Gets / sets the values to be displayed next to the keys. - * If not set, uses data[key][keys[i]]. - * If a function, gets passed currentData (data[key]) and keys[i]. - * @param {*[]} [_=none] - * @returns {tooltip | *[]} - * @memberof tooltip - */ - tooltip.values = function (_) { - return arguments.length ? (values = _, tooltip) : values; - }; - /** - * Gets / sets the header to be displayed in the tooltip. - * If not set, uses key - * @param {string} [_=none] - * @returns {tooltip | string} - * @memberof tooltip - */ - tooltip.header = function (_) { - return arguments.length ? (header = _, tooltip) : header; - }; - /** - * Gets / sets the data (over the selection) to be used for the tooltip - * @param {Object} [_=none] - * @returns {tooltip | Object} - * @memberof tooltip - */ - tooltip.data = function (_) { - return arguments.length ? (data = _, tooltip) : data; - }; - /** - * Gets / sets the selection for the tooltip to be applied on - * @param {d3.selection} [_=none] - * @returns {tooltip | d3.selection} - * @memberof tooltip - */ - tooltip.selection = function (_) { - return arguments.length ? (selection = _, tooltip) : selection; - }; - - /** - * Bind, via selection.on(), the mousemove and mouseout events - * @returns undefined - */ - function tooltip() { - selection.on('mouseover', mousemove); - selection.on('mousemove', mousemove); - selection.on('mouseout', function () { - d3.selectAll(".d3sm-tooltip").remove(); - }); - } - - /** - * Produces the tooltip on mousemove - * @param {string} key of the object targeted by the mousemove - * @param {number} i (index) of the object targeted by mousemove - * @memberof tooltip - * @private - */ - function mousemove(key, i) { - consoleGroup('d3sm-tooltip'); - var currentData = data[key]; - - var _d3$mouse = d3.mouse(d3.select("html").node()), - _d3$mouse2 = slicedToArray(_d3$mouse, 2), - x = _d3$mouse2[0], - y = _d3$mouse2[1]; - - log('tooltip', 'mousemove detected', { key: key, index: i, x: x, y: y }); - log('tooltip', 'current data', currentData); - - var div = safeSelect(d3.select('html'), 'tooltip', 'd3sm-tooltip').classed('card', true).style('max-width', '300px').style('background-color', "#212529").style('color', 'white'); - - var cardBody = safeSelect(div, 'div', 'card-body'); - var cardTitle = safeSelect(cardBody, 'h5', 'card-title').text(header == undefined ? key : typeof header == 'function' ? header(key, currentData, i) : header).style('color', 'cyan'); - - var table = safeSelect(cardBody, 'table', 'table').classed('table-dark', true); - var tBody = safeSelect(table, 'tbody'); - - tBody = tBody.selectAll('tr'); - tBody = tBody.data(keys == undefined ? d3.keys(currentData) : keys); - tBody.exit().remove(); - - var tr = tBody.enter().append('tr').style('max-width', '300px'); - tr.append('td').attr('class', function (d, i) { - return 'tooltip-key'; - }); - tr.append('td').attr('class', function (d, i, j) { - return 'tooltip-value'; - }).attr('tooltip-row-index', function (d, i) { - return i; - }); - - // tBody = tBody.merge(tr) - consoleGroup('tooltip-rows'); - tBody.selectAll('.tooltip-key').text(function (d, i) { - return d; - }); - tBody.selectAll('tr .tooltip-value').text(function (d, i) { - log('tooltip', 'trying to set value', { rowKey: d, rowIndex: i }); - var i = d3.select(this).attr('tooltip-row-index'); - var v = currentData[d]; - - if (values != undefined) { - v = values[i];if (typeof v == "function") { - v = v(currentData, d); - } - } - return typeof v == 'number' ? round(v, 5) : v; - }); - consoleGroupEnd(); - consoleGroupEnd(); - - x += 15; - // x += 15 - var bbox = div.node().getBoundingClientRect(); - if (x + bbox.width > window.innerWidth - window.scrollX) { - x = d3.event.pageX - bbox.width - 15; - } - if (y + bbox.height > window.innerHeight - window.scrollY) { - y = d3.event.pageY - bbox.height - 15; - } - div.style('position') == "relative" ? div.style('position', 'absolute').style('left', x + 'px').style('top', y + 'px') : div.style('left', x + 'px').style('top', y + 'px'); - // .transition().duration(200).ease(d3.easeSin) - - // if (bbox.x + bbox.width > window.innerWidth) { - // div.style('left', (d3.event.pageX-15-bbox.width)+'px') - // } - // if (bbox.y + bbox.height > window.innerHeight) { - // div.style('top', (d3.event.pageY-15-bbox.height)+'px') - // } - - div.attr('z-index', 10000); - } - - return tooltip; - } - - /******************************************************************************* - ** ** - ** ** - ** BAR ** - ** ** - ** ** - *******************************************************************************/ - - /** - * Creates a bar - * - * {@link https://sumneuron.gitlab.io/d3sm/demos/bar-chart-same-data-complex-grouping/index.html Demo} - * @constructor bar - * @param {d3.selection} selection - * @namespace bar - * @returns {function} bar - */ - function bar(selection) { - /* - Assumes that data is list an object. - The keys of data will be bound to the bars. - The valueExtractor function will extract the value from data[key]. - Grouping can be used if desired. It should be an arbitrary complex list where - the values are strings matching keys in data. - */ - var - /** - * Data to plot. Assumed to be a object, where each key corresponds to a bar - * (see {@link bar#data}) - * @param {Object} [data=undefined] - * @memberof bar# - * @property - */ - data, - - /** - * Which direction to render the bars in - * (see {@link bar#orient}) - * @param {number} [orient='horizontal'] - * @memberof bar# - * @property - */ - orient = 'horizontal', - - /** - * Amount of horizontal space (in pixels) avaible to render the bar in - * (see {@link bar#spaceX}) - * @param {number} [spaceX=undefined] - * @memberof bar# - * @property - */ - spaceX, - - /** - * Amount of vertical space (in pixels) avaible to render the bar in - * (see {@link bar.spaceY}) - * @param {number} [spaceY=undefined] - * @memberof bar# - * @property - */ - spaceY, - - - /** - * Whether or not to allow bar to render elements pass the main spatial dimension - * given the orientation (see {@link bar#orient}), where {@link bar#orient}="horizontal" - * the main dimension is {@link bar#spaceX} and where {@link bar#orient}="vertical" - * the main dimension is {@link bar#spaceY} - * @param {boolean} [overflowQ=false] - * @memberof bar# - * @property - */ - overflowQ = false, - - - /** - * An array - putatively of other arrays - depicting how bars should be arranged - * @param {Array[]} [grouping=undefined] - * @memberof bar# - * @property - */ - grouping, - - - /** - * How to get the value of the bar - * @param {function} [valueExtractor=function(key, index) { return data[key] }] - * @memberof bar# - * @property - */ - valueExtractor = function valueExtractor(key, index) { - return data[key]; - }, - - /** - * How to sort the bars - if {@link bar#grouping} is not provided. - * @param {function} [sortingFunction=function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}] - * @memberof bar# - * @property - */ - sortingFunction = function sortingFunction(keyA, keyB) { - return d3.descending(data[keyA], data[keyB]); - }, - - - /** - * The scale for which bar values should be transformed by - * @param {d3.scale} [scale=d3.scaleLinear] - * @memberof bar# - * @property - */ - scale = d3.scaleLinear(), - - /** - * The padding for the domain of the scale (see {@link bar#scale}) - * @param {number} [domainPadding=0.5] - * @memberof bar# - * @property - */ - domainPadding = 0.5, - - - /** - * Default space for the spacer (percentage) of main dimension given the orientation - * (see {@link bar#orient}), where {@link bar#orient}="horizontal" - * the main dimension is {@link bar#spaceX} and where {@link bar#orient}="vertical" - * the main dimension is {@link bar#spaceY} between bars - * @param {number} [objectSpacer=0.05] - * @memberof bar# - * @property - */ - objectSpacer = 0.05, - - /** - * The minimum size that an object can be - * @param {number} [minObjectSize=50] - * @memberof bar# - * @property - */ - minObjectSize = 50, - - /** - * The maximum size that an object can be - * @param {number} [maxObjectSize=100] - * @memberof bar# - * @property - */ - maxObjectSize = 100, - - - /** - * The stroke width of the bars - * @param {number} [barStrokeWidth=2] - * @memberof bar# - * @property - */ - barStrokeWidth = 2, - - /** - * Instance of ColorFunction - * @param {function} [colorFunction = colorFunction()] - * @memberof bar# - * @property - */ - colorFunction$$1 = colorFunction(), - - - /** - * Color of the background - * @param {string} [backgroundFill="transparent"] - * @memberof bar# - * @property - */ - backgroundFill = 'transparent', - - /** - * Namespace for all items made by this instance of bar - * @param {string} [namespace="d3sm-bar"] - * @memberof bar# - * @property - */ - namespace = 'd3sm-bar', - - /** - * Class name for bar container ( element) - * @param {string} [objectClass="bar"] - * @memberof bar# - * @property - */ - objectClass = 'bar', - - - /** - * Duration of all transitions of this element - * @param {number} [transitionDuration=1000] - * @memberof bar# - * @property - */ - transitionDuration = 1000, - - /** - * Easing function for transitions - * @param {d3.ease} [easeFunc=d3.easeExp] - * @memberof bar# - * @property - */ - easeFunc = d3.easeExp, - - - // useful values to extract to prevent re-calculation - /** - * The keys of the bars - * @param {string[]} [barKeys=undefined] - * @memberof bar# - * @property - */ - barKeys, - - /** - * The values of the bars - * @param {number[]} [barValues=undefined] - * @memberof bar# - * @property - */ - barValues, - - /** - * The objectSize (actual width) used by the bars - * @param {number} [objectSize=undefined] - * @memberof bar# - * @property - */ - objectSize, - - /** - * The spacerSize (actual width) used by the spacers between the bars - * @param {number} [spacerSize=undefined] - * @memberof bar# - * @property - */ - spacerSize, - - /** - * Instance of Tooltip - * @param {function} [tooltip=tooltip()] - * @memberof bar# - * @property - */ - tooltip$$1 = tooltip(), - barPercent = 1; - - /** - * Gets or sets the selection in which items are manipulated - * @param {d3.selection} [_=none] - * @returns {bar | d3.selection} - * @memberof bar - * @property - * by default selection = selection - */ - bar.selection = function (_) { - return arguments.length ? (selection = _, bar) : selection; - }; - /** - * Gets or sets the data - * (see {@link bar#data}) - * @param {number} [_=none] - * @returns {bar | object} - * @memberof bar - * @property - */ - bar.data = function (_) { - return arguments.length ? (data = _, bar) : data; - }; - /** - * Gets or sets the orient of the bars - * (see {@link bar#orient}) - * @param {number} [_=none] - * @returns {bar | object} - * @memberof bar - * @property - */ - bar.orient = function (_) { - return arguments.length ? (orient = _, bar) : orient; - }; - /** - * Gets or sets the amount of horizontal space in which items are manipulated - * (see {@link bar#spaceX}) - * @param {number} [_=none] should be a number > 0 - * @returns {bar | number} - * @memberof bar - * @property - * by default spaceX = undefined - */ - bar.spaceX = function (_) { - return arguments.length ? (spaceX = _, bar) : spaceX; - }; - /** - * Gets or sets the amount of vertical space in which items are manipulated - * (see {@link bar#spaceY}) - * @param {number} [_=none] should be a number > 0 - * @returns {bar | number} - * @memberof bar - * @property - * by default spaceY = undefined - */ - bar.spaceY = function (_) { - return arguments.length ? (spaceY = _, bar) : spaceY; - }; - - /** - * Gets / sets whether or not bar is allowed to go beyond specified dimensions - * (see {@link bar#spaceX}) - * @param {boolean} [_=none] - * @returns {bar | boolean} - * @memberof bar - * @property - * by default overflowQ = false - */ - bar.overflowQ = function (_) { - return arguments.length ? (overflowQ = _, bar) : overflowQ; - }; - /** - * Gets / sets the grouping of the bars - * (see {@link bar#grouping}) - * @param {Array[]} [_=none] - * @returns {bar | Array[]} - * @memberof bar - * @property - * by default grouping = undefined - */ - bar.grouping = function (_) { - return arguments.length ? (grouping = _, bar) : grouping; - }; - /** - * Gets / sets the valueExtractor - * (see {@link bar#valueExtractor}) - * @param {function} [_=none] - * @returns {bar | function} - * @memberof bar - * @property - * by default valueExtractor = function(key, index) { return data[key] }, - */ - bar.valueExtractor = function (_) { - return arguments.length ? (valueExtractor = _, bar) : valueExtractor; - }; - /** - * Gets / sets the sortingFunction - * (see {@link bar#sortingFunction}) - * @param {function} [_=none] - * @returns {bar | function} - * @memberof bar - * @property - * by default sortingFunction = function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}, - */ - bar.sortingFunction = function (_) { - return arguments.length ? (sortingFunction = _, bar) : sortingFunction; - }; - /** - * Gets / sets the scale for which the bar values should be transformed by - * (see {@link bar#scale}) - * @param {d3.scale} [_=none] - * @returns {bar | d3.scale} - * @memberof bar - * @property - * by default scale = d3.scaleLinear() - */ - bar.scale = function (_) { - return arguments.length ? (scale = _, bar) : scale; - }; - /** - * Gets / sets the padding for the domain of the scale - * (see {@link bar#domainPadding}) - * @param {number} [_=none] - * @returns {bar | number} - * @memberof bar - * @property - * by default domainPadding = 0.5 - */ - bar.domainPadding = function (_) { - return arguments.length ? (domainPadding = _, bar) : domainPadding; - }; - /** - * Gets / sets objectSpacer - * (see {@link bar#objectSpacer}) - * @param {number} [_=none] - * @returns {bar | number} - * @memberof bar - * @property - * by default objectSpacer = 0.05 - */ - bar.objectSpacer = function (_) { - return arguments.length ? (objectSpacer = _, bar) : objectSpacer; - }; - /** - * Gets / sets the minObjectSize - * (see {@link bar#minObjectSize}) - * @param {number} [_=none] - * @returns {bar | number} - * @memberof bar - * @property - * by default minObjectSize = 50 - */ - bar.minObjectSize = function (_) { - return arguments.length ? (minObjectSize = _, bar) : minObjectSize; - }; - /** - * Gets / sets the maxObjectSize - * (see {@link bar#maxObjectSize}) - * @param {number} [_=none] - * @returns {bar | number} - * @memberof bar - * @property - * by default maxObjectSize = 100 - */ - bar.maxObjectSize = function (_) { - return arguments.length ? (maxObjectSize = _, bar) : maxObjectSize; - }; - - /** - * Gets / sets the barStrokeWidth - * (see {@link bar#barStrokeWidth}) - * @param {number} [_=none] - * @returns {bar | number} - * @memberof bar - * @property - * by default barStrokeWidth = 2 - */ - bar.barStrokeWidth = function (_) { - return arguments.length ? (barStrokeWidth = _, bar) : barStrokeWidth; - }; - /** - * Gets / sets the colorFunction - * (see {@link bar#colorFunction}) - * @param {number} [_=none] - * @returns {bar | number} - * @memberof bar - * @property - * by default colorFunction = colorFunction() - */ - bar.colorFunction = function (_) { - return arguments.length ? (colorFunction$$1 = _, bar) : colorFunction$$1; - }; - - /** - * Gets / sets the backgroundFill - * (see {@link bar#backgroundFill}) - * @param {string} [_=none] - * @returns {bar | string} - * @memberof bar - * @property - * by default backgroundFill = 'transparent' - */ - bar.backgroundFill = function (_) { - return arguments.length ? (backgroundFill = _, bar) : backgroundFill; - }; - /** - * Gets / sets the namespace - * (see {@link bar#namespace}) - * @param {string} [_=none] - * @returns {bar | string} - * @memberof bar - * @property - * by default namespace = 'd3sm-bar' - */ - bar.namespace = function (_) { - return arguments.length ? (namespace = _, bar) : namespace; - }; - /** - * Gets / sets the objectClass - * (see {@link bar#objectClass}) - * @param {string} [_=none] - * @returns {bar | string} - * @memberof bar - * @property - * by default objectClass = 'tick-group' - */ - bar.objectClass = function (_) { - return arguments.length ? (objectClass = _, bar) : objectClass; - }; - /** - * Gets / sets the transitionDuration - * (see {@link bar#transitionDuration}) - * @param {number} [_=none] - * @returns {bar | number} - * @memberof bar - * @property - * by default transitionDuration = 1000 - */ - bar.transitionDuration = function (_) { - return arguments.length ? (transitionDuration = _, bar) : transitionDuration; - }; - /** - * Gets / sets the easeFunc - * (see {@link bar#easeFunc}) - * @param {d3.ease} [_=none] - * @returns {bar | d3.ease} - * @memberof bar - * @property - * by default easeFunc = d3.easeExp - */ - bar.easeFunc = function (_) { - return arguments.length ? (easeFunc = _, bar) : easeFunc; - }; - - /** - * Gets / sets the barKeys - * (see {@link bar#barKeys}) - * @param {string[]} [_=none] - * @returns {bar | string[]} - * @memberof bar - * @property - * by default barKeys = undefined - */ - bar.barKeys = function (_) { - return arguments.length ? (barKeys = _, bar) : barKeys; - }; - /** - * Gets / sets the barValues - * (see {@link bar#barValues}) - * @param {number[]} [_=none] - * @returns {bar | number[]} - * @memberof bar - * @property - * by default barValues = undefined - */ - bar.barValues = function (_) { - return arguments.length ? (barValues = _, bar) : barValues; - }; - /** - * Gets / sets the objectSize - * (see {@link bar#objectSize}) - * @param {number} [_=none] - * @returns {bar | number} - * @memberof bar - * @property - * by default objectSize = undefined - */ - bar.objectSize = function (_) { - return arguments.length ? (objectSize = _, bar) : objectSize; - }; - /** - * Gets / sets the spacerSize - * (see {@link bar#spacerSize}) - * @param {number} [_=none] - * @returns {bar | number} - * @memberof bar - * @property - * by default spacerSize = undefined - */ - bar.spacerSize = function (_) { - return arguments.length ? (spacerSize = _, bar) : spacerSize; - }; - - /** - * Gets / sets the tooltip - * (see {@link bar#tooltip}) - * @param {tooltip} [_=none] - * @returns {bar | tooltip} - * @memberof bar - * @property - * by default tooltip = tooltip() - */ - bar.tooltip = function (_) { - return arguments.length ? (tooltip$$1 = _, bar) : tooltip$$1; - }; - - bar.barPercent = function (_) { - return arguments.length ? (barPercent = _, bar) : barPercent; - }; - - function bar() { - // for convenience in handling orientation specific values - var horizontalQ = orient == 'horizontal' || orient == 'bottom' || orient == 'top' ? true : false; - var verticalQ = !horizontalQ; - - // background cliping rectangle - var bgcpRect = { x: 0, y: 0, width: spaceX, height: spaceY }; - var container = setupContainer(selection, namespace, bgcpRect, backgroundFill); - - // to prevent re-calculation and getters to be passed to axes - barKeys = d3.keys(data); - barValues = barKeys.map(valueExtractor); - - // if grouping is undefined sort barKeys by sortingFunction - var ordered = grouping == undefined ? barKeys.sort(sortingFunction) : grouping; - // ordered might be nested depending on grouping - barKeys = flatten(ordered); - - var numberOfObjects = barKeys.length; - var extent = [Math.min.apply(Math, toConsumableArray(barValues)) - domainPadding, Math.max.apply(Math, toConsumableArray(barValues)) + domainPadding]; - - // set the scale - - scale.domain(extent).range(horizontalQ ? [0, spaceY] : orient == 'right' ? [0, spaceX] : [spaceX, 0]); - var space = horizontalQ ? spaceX : spaceY; - // calculate object size - objectSize = objectSize == undefined ? calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ) : objectSize; - - // calculate spacer size if needed - spacerSize = spacerSize == undefined ? calculateWidthOfSpacer(barKeys, space, objectSize, numberOfObjects, objectSpacer, overflowQ) : spacerSize; - // make the nested groups - var spacerFunction = groupingSpacer().horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects).objectClass(objectClass).objectSize(objectSize).spacerSize(spacerSize).transitionDuration(transitionDuration).easeFunc(easeFunc).namespace(namespace); - // safe default function - var defaultExit = spacerFunction.exitFunction(); - - spacerFunction.exitFunction(function (sel) { - // use default to move objects off screen - // console.log("EXIT", sel.nodes(), objectSize, scale(extent[1])) - if (objectSize == undefined) { - console.log(sel.nodes(), objectSize); - } - defaultExit(sel); - sel.selectAll('g').classed("to-remove", true); - // shrink rectangles in addition - sel.selectAll('* > rect').transition().duration(transitionDuration).attr('transform', function (d, i) { - var x = horizontalQ ? 0 : 0, - y = verticalQ ? 0 : scale(extent[1]), - t = 'translate(' + x + ',' + y + ')'; - return t; - }).attr('width', horizontalQ ? objectSize : 0).attr('height', verticalQ ? objectSize : 0).remove(); - }); - - // move stuff - spacerFunction(container, ordered, 0); - - var parentIndexArray = []; - container.selectAll('g:not(.to-remove).' + objectClass).each(function (d, i) { - parentIndexArray.push(Number(d3.select(this).attr('parent-index'))); - }); - - colorFunction$$1 = colorFunction$$1.colorBy() == 'index' ? colorFunction$$1.dataExtent([0, Math.max.apply(Math, parentIndexArray)]) : colorFunction$$1.dataExtent(extent); - - container.selectAll('g.' + objectClass + ':not(.to-remove)').each(function (key, i) { - // console.log(key, scale(extent[1]) - scale(valueExtractor(key, i))) - var t = d3.select(this), - currentData = data[key], - value = valueExtractor(key, i), - i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'), - fillColor = colorFunction$$1(key, value, i, 'fill'), - // prevent duplicate computation - strokeColor = colorFunction$$1(key, value, i, 'stroke'); - - var bar = safeSelect(t, 'rect', 'bar-rect'); - - if (bar.attr('transform') == undefined) { - bar.attr('transform', function (d, i) { - var x = horizontalQ ? 0 : 0, - y = verticalQ ? 0 : scale(extent[1]), - t = 'translate(' + x + ',' + y + ')'; - return t; - }).attr('width', horizontalQ ? objectSize : 0).attr('height', verticalQ ? objectSize : 0); - } - - bar.transition().duration(transitionDuration).ease(easeFunc).attr('transform', function (d, i) { - var x = horizontalQ ? objectSize - objectSize * barPercent : orient == 'right' ? scale(extent[1]) - scale(value) : objectSize - objectSize * barPercent, - y = verticalQ ? objectSize - objectSize * barPercent : scale(extent[1]) - scale(value), - t = 'translate(' + x + ',' + y + ')'; - return t; - }).attr('width', horizontalQ ? objectSize * barPercent : scale(value)).attr('height', verticalQ ? objectSize * barPercent : scale(value)).attr('fill', fillColor).attr('stroke', strokeColor).attr('stroke-width', barStrokeWidth); - - t.on('mouseover', function (d, i) { - container.selectAll('g.' + objectClass).style('opacity', 0.2); - t.style('opacity', 1); - bar.attr('stroke-width', barStrokeWidth * 2); - }); - t.on('mouseout', function () { - container.selectAll('g.' + objectClass).style('opacity', 1); - bar.attr('stroke-width', barStrokeWidth); - }); - }); - - tooltip$$1.selection(container.selectAll('.bar-rect')).data(data); - - tooltip$$1(); - } - return bar; - } - - /** - * Creates a bubbleHeatmap - * - * {@link https://sumneuron.gitlab.io/d3sm/demos/bubble-heatmap/index.html Demo} - * @constructor bubbleHeatmap - * @param {d3.selection} selection - * @namespace bubbleHeatmap - * @returns {function} bubbleHeatmap - */ - function bubbleHeatmap(selection) { - var - /** - * Data to plot. Assumed to be a object, where each key corresponds to a cell - * (see {@link bubbleHeatmap#data}) - * @param {Object} [data=undefined] - * @memberof bubbleHeatmap# - * @property - */ - data, - - - /** - * Amount of horizontal space (in pixels) avaible to render the bubbleHeatmap in - * (see {@link bubbleHeatmap#spaceX}) - * @param {number} [spaceX=undefined] - * @memberof bubbleHeatmap# - * @property - */ - spaceX, - - /** - * Amount of vertical space (in pixels) avaible to render the bubbleHeatmap in - * (see {@link bubbleHeatmap.spaceY}) - * @param {number} [spaceY=undefined] - * @memberof bubbleHeatmap# - * @property - */ - spaceY, - - - /** - * The internal key of the cell specifiying to which x axis key it belongs - * (see {@link bubbleHeatmap.xKey}) - * @param {string} [xKey='x'] - * @memberof bubbleHeatmap# - * @property - */ - xKey = 'x', - - /** - * The internal key of the cell specifiying to which y axis key it belongs - * (see {@link bubbleHeatmap.yKey}) - * @param {string} [yKey='y'] - * @memberof bubbleHeatmap# - * @property - */ - yKey = 'y', - - /** - * The internal key of the cell specifiying what value to use to determine the radius - * (see {@link bubbleHeatmap.rKey}) - * @param {string} [rKey='r'] - * @memberof bubbleHeatmap# - * @property - */ - rKey = 'r', - - /** - * The internal key of the cell specifiying what value to use to determine the color - * (see {@link bubbleHeatmap.vKey}) - * @param {string} [vKey='v'] - * @memberof bubbleHeatmap# - * @property - */ - vKey = 'v', - - - /** - * Function for extracting the the value from xKey. - * (see {@link bubbleHeatmap.xExtractor}) - * @param {function} [xExtractor=function(key, i) { return data[key][xKey] }] - * @returns {string} - * @memberof bubbleHeatmap# - * @property - */ - xExtractor = function xExtractor(key, i) { - return data[key][xKey]; - }, - - /** - * Function for extracting the the value from yKey. - * (see {@link bubbleHeatmap.yExtractor}) - * @param {function} [yExtractor=function(key, i) { return data[key][yKey] }] - * @returns {string} - * @memberof bubbleHeatmap# - * @property - */ - yExtractor = function yExtractor(key, i) { - return data[key][yKey]; - }, - - /** - * Function for extracting the the value from rKey. - * (see {@link bubbleHeatmap.rExtractor}) - * @param {function} [rExtractor=function(key, i) { return data[key][rKey] }] - * @returns {number} - * @memberof bubbleHeatmap# - * @property - */ - rExtractor = function rExtractor(key, i) { - return data[key][rKey]; - }, - - /** - * Function for extracting the the value from vKey. - * (see {@link bubbleHeatmap.vExtractor}) - * @param {function} [vExtractor=function(key, i) { return data[key][vKey] }] - * @returns {number} - * @memberof bubbleHeatmap# - * @property - */ - vExtractor = function vExtractor(key, i) { - return data[key][vKey]; - }, - - - /** - * Whether or not to allow bubbleHeatmap to render elements pass the main spatial dimension - * given the orientation (see {@link bubbleHeatmap#orient}), where {@link bubbleHeatmap#orient}="bottom" or {@link bubbleHeatmap#orient}="top" - * the main dimension is {@link bubbleHeatmap#spaceX} and where {@link bubbleHeatmap#orient}="left" or {@link bubbleHeatmap#orient}="right" - * the main dimension is {@link bubbleHeatmap#spaceY} - * @param {boolean} [overflowQ=false] - * @memberof bubbleHeatmap# - * @property - */ - overflowQ = false, - - - /** - * The scale for which the radius values should be transformed by - * @param {d3.scale} [scale=d3.scaleLinear] - * @memberof bubbleHeatmap# - * @property - */ - scale = d3.scaleLinear(), - - /** - * The padding for the domain of the scale (see {@link bubbleHeatmap#scale}) - * @param {number} [domainPadding=0.5] - * @memberof bubbleHeatmap# - * @property - */ - domainPadding = 0.5, - - - /** - * Default space for the spacer (percentage) of main dimension given the orientation - * (see {@link bubbleHeatmap#orient}), where {@link bubbleHeatmap#orient}="horizontal" - * the main dimension is {@link bubbleHeatmap#spaceX} and where {@link bubbleHeatmap#orient}="vertical" - * the main dimension is {@link bubbleHeatmap#spaceY} between bubbles - * @param {number} [objectSpacer=0.0] - * @memberof bubbleHeatmap# - * @property - */ - objectSpacer = 0.0, - - /** - * The minimum size that an object can be - * @param {number} [minObjectSize=50] - * @memberof bubbleHeatmap# - * @property - */ - minObjectSize = 50, - - /** - * The maximum size that an object can be - * @param {number} [maxObjectSize=100] - * @memberof bubbleHeatmap# - * @property - */ - maxObjectSize = 100, - - - /** - * The stroke width of the bubbles - * @param {number} [bubbleStrokeWidth=2] - * @memberof bubbleHeatmap# - * @property - */ - bubbleStrokeWidth = 2, - - // colorFunc = colorFunction(), - - /** - * Color of the background - * @param {string} [backgroundFill="transparent"] - * @memberof bubbleHeatmap# - * @property - */ - backgroundFill = 'transparent', - - /** - * Namespace for all items made by this instance of bubbleHeatmap - * @param {string} [namespace="d3sm-bubble"] - * @memberof bubbleHeatmap# - * @property - */ - namespace = 'd3sm-bubble', - - /** - * Class name for bubble container ( element) - * @param {string} [objectClass="bubble"] - * @memberof bubbleHeatmap# - * @property - */ - objectClass = 'bubble', - - /** - * Duration of all transitions of this element - * @param {number} [transitionDuration=1000] - * @memberof bubbleHeatmap# - * @property - */ - transitionDuration = 1000, - - /** - * Easing function for transitions - * @param {d3.ease} [easeFunc=d3.easeExp] - * @memberof bubbleHeatmap# - * @property - */ - easeFunc = d3.easeExp, - - - /** - * Stores the keys of all the cells - * Calculated after bubbleHeatmap called. - * @param {string[]} [cellKeys=undefined] - * @memberof bubbleHeatmap# - * @property - */ - cellKeys, - - /** - * Stores the list of unique xValues - * Calculated after bubbleHeatmap called. - * @param {string[]} [xValues=undefined] - * @memberof bubbleHeatmap# - * @property - */ - xValues, - - /** - * Stores the list of unique yValues - * Calculated after bubbleHeatmap called. - * @param {string[]} [yValues=undefined] - * @memberof bubbleHeatmap# - * @property - */ - yValues, - - /** - * Stores the list of unique rValues - * Calculated after bubbleHeatmap called. - * @param {string[]} [rValues=undefined] - * @memberof bubbleHeatmap# - * @property - */ - rValues, - - /** - * Stores the list of unique vValues - * Calculated after bubbleHeatmap called. - * @param {string[]} [vValues=undefined] - * @memberof bubbleHeatmap# - * @property - */ - vValues, - xKeySortingFunction = function xKeySortingFunction(a, b) { - return xExtractor(a) - xExtractor(b); - }, - yKeySortingFunction = function yKeySortingFunction(a, b) { - return yExtractor(a) - yExtractor(b); - }, - /** - * Instance of ColorFunction with .colorBy set to 'category' - * @function colorFunction - * @memberof bubbleHeatmap# - * @property - */ - colorFunction$$1 = colorFunction().colorBy('value'), - - /** - * Instance of Tooltip - * @function tooltip - * @memberof bubbleHeatmap# - * @property - */ - tooltip$$1 = tooltip(), - - - /** - * store the size the bubble could be in the x dimension - * the actuall size of the bubble will be the min of xSize and {@link bubbleHeatmap#ySize} - * Calculated after bubbleHeatmap called. - * @param {string[]} [xSize=undefined] - * @memberof bubbleHeatmap# - * @property - */ - xSize, - - /** - * store the size of the spacer in the x dimension - * Calculated after bubbleHeatmap called. - * @param {string[]} [xSpacerSize=undefined] - * @memberof bubbleHeatmap# - * @property - */ - xSpacerSize, - - - /** - * store the size the bubble could be in the y dimension - * the actuall size of the bubble will be the min of xSize and {@link bubbleHeatmap#xSize} - * Calculated after bubbleHeatmap called. - * @param {string[]} [ySize=undefined] - * @memberof bubbleHeatmap# - * @property - */ - ySize, - - /** - * store the size of the spacer in the y dimension. - * Calculated after bubbleHeatmap called. - * @param {string[]} [xSpacerSize=undefined] - * @memberof bubbleHeatmap# - * @property - */ - ySpacerSize; - - /** - * Gets or sets the selection in which items are manipulated - * @param {d3.selection} [_=none] - * @returns {bubbleHeatmap | d3.selection} - * @memberof bubbleHeatmap - * @property - * by default selection = selection - */ - bhm.selection = function (_) { - return arguments.length ? (selection = _, bhm) : selection; - }; - /** - * Gets or sets the data - * (see {@link bubbleHeatmap#data}) - * @param {number} [_=none] - * @returns {bubbleHeatmap | object} - * @memberof bubbleHeatmap - * @property - */ - bhm.data = function (_) { - return arguments.length ? (data = _, bhm) : data; - }; - // bhm.orient = function(_) { return arguments.length ? (orient = _, bhm) : orient; } - /** - * Gets or sets the amount of horizontal space in which items are manipulated - * (see {@link bubbleHeatmap#spaceX}) - * @param {number} [_=none] should be a number > 0 - * @returns {bubbleHeatmap | number} - * @memberof bubbleHeatmap - * @property - * by default spaceX = undefined - */ - bhm.spaceX = function (_) { - return arguments.length ? (spaceX = _, bhm) : spaceX; - }; - /** - * Gets or sets the amount of vertical space in which items are manipulated - * (see {@link bubbleHeatmap#spaceY}) - * @param {number} [_=none] should be a number > 0 - * @returns {bubbleHeatmap | number} - * @memberof bubbleHeatmap - * @property - * by default spaceY = undefined - */ - bhm.spaceY = function (_) { - return arguments.length ? (spaceY = _, bhm) : spaceY; - }; - - /** - * Gets or sets the xKey - * (see {@link bubbleHeatmap#xKey}) - * @param {string} [_=none] - * @returns {bubbleHeatmap | string} - * @memberof bubbleHeatmap - * @property - * by default xKey = 'x' - */ - bhm.xKey = function (_) { - return arguments.length ? (xKey = _, bhm) : xKey; - }; - /** - * Gets or sets the yKey - * (see {@link bubbleHeatmap#yKey}) - * @param {string} [_=none] - * @returns {bubbleHeatmap | string} - * @memberof bubbleHeatmap - * @property - * by default yKey = 'y' - */ - bhm.yKey = function (_) { - return arguments.length ? (yKey = _, bhm) : yKey; - }; - /** - * Gets or sets the rKey - * (see {@link bubbleHeatmap#rKey}) - * @param {string} [_=none] - * @returns {bubbleHeatmap | string} - * @memberof bubbleHeatmap - * @property - * by default rKey = 'r' - */ - bhm.rKey = function (_) { - return arguments.length ? (rKey = _, bhm) : rKey; - }; - /** - * Gets or sets the vKey - * (see {@link bubbleHeatmap#vKey}) - * @param {string} [_=none] - * @returns {bubbleHeatmap | string} - * @memberof bubbleHeatmap - * @property - * by default vKey = 'y' - */ - bhm.vKey = function (_) { - return arguments.length ? (vKey = _, bhm) : vKey; - }; - - /** - * Gets or sets the cellKeys - * (see {@link bubbleHeatmap#cellKeys}) - * @param {string[]} [_=none] - * @returns {bubbleHeatmap | string[]} - * @memberof bubbleHeatmap - * @property - * by default cellKeys = undefined - */ - bhm.cellKeys = function (_) { - return arguments.length ? (cellKeys = _, bhm) : cellKeys; - }; - /** - * Gets or sets the xValues - * (see {@link bubbleHeatmap#xValues}) - * @param {string[]} [_=none] - * @returns {bubbleHeatmap | string[]} - * @memberof bubbleHeatmap - * @property - * by default xValues = undefined - */ - bhm.xValues = function (_) { - return arguments.length ? (xValues = _, bhm) : xValues; - }; - /** - * Gets or sets the yValues - * (see {@link bubbleHeatmap#yValues}) - * @param {string[]} [_=none] - * @returns {bubbleHeatmap | string[]} - * @memberof bubbleHeatmap - * @property - * by default yValues = undefined - */ - bhm.yValues = function (_) { - return arguments.length ? (yValues = _, bhm) : yValues; - }; - /** - * Gets or sets the rValues - * (see {@link bubbleHeatmap#rValues}) - * @param {number[]} [_=none] - * @returns {bubbleHeatmap | number[]} - * @memberof bubbleHeatmap - * @property - * by default rValues = undefined - */ - bhm.rValues = function (_) { - return arguments.length ? (rValues = _, bhm) : rValues; - }; - /** - * Gets or sets the vValues - * (see {@link bubbleHeatmap#vValues}) - * @param {number[]} [_=none] - * @returns {bubbleHeatmap | number[]} - * @memberof bubbleHeatmap - * @property - * by default vValues = undefined - */ - bhm.vValues = function (_) { - return arguments.length ? (vValues = _, bhm) : vValues; - }; - - /** - * Gets or sets the xExtractor - * (see {@link bubbleHeatmap#xExtractor}) - * @param {function} [_=none] - * @returns {bubbleHeatmap | function} - * @memberof bubbleHeatmap - * @property - * by default xExtractor = undefined - */ - bhm.xExtractor = function (_) { - return arguments.length ? (xExtractor = _, bhm) : xExtractor; - }; - /** - * Gets or sets the yExtractor - * (see {@link bubbleHeatmap#yExtractor}) - * @param {function} [_=none] - * @returns {bubbleHeatmap | function} - * @memberof bubbleHeatmap - * @property - * by default yExtractor = undefined - */ - bhm.yExtractor = function (_) { - return arguments.length ? (yExtractor = _, bhm) : yExtractor; - }; - /** - * Gets or sets the rExtractor - * (see {@link bubbleHeatmap#rExtractor}) - * @param {function} [_=none] - * @returns {bubbleHeatmap | function} - * @memberof bubbleHeatmap - * @property - * by default rExtractor = undefined - */ - bhm.rExtractor = function (_) { - return arguments.length ? (rExtractor = _, bhm) : rExtractor; - }; - /** - * Gets or sets the vExtractor - * (see {@link bubbleHeatmap#vExtractor}) - * @param {function} [_=none] - * @returns {bubbleHeatmap | function} - * @memberof bubbleHeatmap - * @property - * by default vExtractor = undefined - */ - bhm.vExtractor = function (_) { - return arguments.length ? (vExtractor = _, bhm) : vExtractor; - }; - - /** - * Gets / sets whether or not bubbleHeatmap is allowed to go beyond specified dimensions - * (see {@link bubbleHeatmap#spaceX}) - * @param {boolean} [_=none] - * @returns {bubbleHeatmap | boolean} - * @memberof bubbleHeatmap - * @property - * by default overflowQ = false - */ - bhm.overflowQ = function (_) { - return arguments.length ? (overflowQ = _, bhm) : overflowQ; - }; - /** - * Gets / sets the scale for which the radius of bubbles should be transformed by - * (see {@link bubbleHeatmap#scale}) - * @param {d3.scale} [_=none] - * @returns {bubbleHeatmap | d3.scale} - * @memberof bubbleHeatmap - * @property - * by default scale = d3.scaleLinear() - */ - bhm.scale = function (_) { - return arguments.length ? (scale = _, bhm) : scale; - }; - /** - * Gets / sets the padding for the domain of the scale - * (see {@link bubbleHeatmap#domainPadding}) - * @param {number} [_=none] - * @returns {bubbleHeatmap | number} - * @memberof bubbleHeatmap - * @property - * by default domainPadding = 0.5 - */ - bhm.domainPadding = function (_) { - return arguments.length ? (domainPadding = _, bhm) : domainPadding; - }; - /** - * Gets / sets objectSpacer - * (see {@link bubbleHeatmap#objectSpacer}) - * @param {number} [_=none] - * @returns {bubbleHeatmap | number} - * @memberof bubbleHeatmap - * @property - * by default objectSpacer = 0.0 - */ - bhm.objectSpacer = function (_) { - return arguments.length ? (objectSpacer = _, objectSpacer) : data; - }; - /** - * Gets / sets the minObjectSize - * (see {@link bubbleHeatmap#minObjectSize}) - * @param {number} [_=none] - * @returns {bubbleHeatmap | number} - * @memberof bubbleHeatmap - * @property - * by default minObjectSize = 50 - */ - bhm.minObjectSize = function (_) { - return arguments.length ? (minObjectSize = _, bhm) : minObjectSize; - }; - /** - * Gets / sets the maxObjectSize - * (see {@link bubbleHeatmap#maxObjectSize}) - * @param {number} [_=none] - * @returns {bubbleHeatmap | number} - * @memberof bubbleHeatmap - * @property - * by default maxObjectSize = 100 - */ - bhm.maxObjectSize = function (_) { - return arguments.length ? (maxObjectSize = _, bhm) : maxObjectSize; - }; - /** - * Gets / sets the bubbleStrokeWidth - * (see {@link bubbleHeatmap#bubbleStrokeWidth}) - * @param {number} [_=none] - * @returns {bubbleHeatmap | number} - * @memberof bubbleHeatmap - * @property - * by default bubbleStrokeWidth = 2 - */ - bhm.bubbleStrokeWidth = function (_) { - return arguments.length ? (bubbleStrokeWidth = _, bhm) : bubbleStrokeWidth; - }; - /** - * Gets / sets the backgroundFill - * (see {@link bubbleHeatmap#backgroundFill}) - * @param {string} [_=none] - * @returns {bubbleHeatmap | string} - * @memberof bubbleHeatmap - * @property - * by default backgroundFill = 'transparent' - */ - bhm.backgroundFill = function (_) { - return arguments.length ? (backgroundFill = _, bhm) : backgroundFill; - }; - /** - * Gets / sets the namespace - * (see {@link bubbleHeatmap#namespace}) - * @param {string} [_=none] - * @returns {bubbleHeatmap | string} - * @memberof bubbleHeatmap - * @property - * by default namespace = 'd3sm-bubbleHeatmap' - */ - bhm.namespace = function (_) { - return arguments.length ? (namespace = _, bhm) : namespace; - }; - /** - * Gets / sets the objectClass - * (see {@link bubbleHeatmap#objectClass}) - * @param {string} [_=none] - * @returns {bubbleHeatmap | string} - * @memberof bubbleHeatmap - * @property - * by default objectClass = 'tick-group' - */ - bhm.objectClass = function (_) { - return arguments.length ? (objectClass = _, bhm) : objectClass; - }; - /** - * Gets / sets the transitionDuration - * (see {@link bubbleHeatmap#transitionDuration}) - * @param {number} [_=none] - * @returns {bubbleHeatmap | number} - * @memberof bubbleHeatmap - * @property - * by default transitionDuration = 1000 - */ - bhm.transitionDuration = function (_) { - return arguments.length ? (transitionDuration = _, bhm) : transitionDuration; - }; - /** - * Gets / sets the easeFunc - * (see {@link bubbleHeatmap#easeFunc}) - * @param {d3.ease} [_=none] - * @returns {bubbleHeatmap | d3.ease} - * @memberof bubbleHeatmap - * @property - * by default easeFunc = d3.easeExp - */ - bhm.easeFunc = function (_) { - return arguments.length ? (easeFunc = _, bhm) : easeFunc; - }; - - /** - * Gets / sets the tooltip - * (see {@link bubbleHeatmap#tooltip}) - * @param {tooltip} [_=none] - * @returns {bubbleHeatmap | tooltip} - * @memberof bubbleHeatmap - * @property - * by default tooltip = tooltip() - */ - bhm.tooltip = function (_) { - return arguments.length ? (tooltip$$1 = _, bhm) : tooltip$$1; - }; - - /** - * Gets / sets the colorFunction - * (see {@link bubbleHeatmap#colorFunction}) - * @param {colorFunction} [_=none] - * @returns {bubbleHeatmap | colorFunction} - * @memberof bubbleHeatmap - * @property - * by default colorFunction = colorFunction() - */ - bhm.colorFunction = function (_) { - return arguments.length ? (colorFunction$$1 = _, bhm) : colorFunction$$1; - }; - - /** - * Gets / sets the xSize - * (see {@link bubbleHeatmap#xSize}) - * @param {number} [_=none] - * @returns {bubbleHeatmap | number} - * @memberof bubbleHeatmap - * @property - * by default xSize = undefined - */ - bhm.xSize = function (_) { - return arguments.length ? (xSize = _, bhm) : xSize; - }; - /** - * Gets / sets the xSpacerSize - * (see {@link bubbleHeatmap#xSpacerSize}) - * @param {number} [_=none] - * @returns {bubbleHeatmap | number} - * @memberof bubbleHeatmap - * @property - * by default xSpacerSize = undefined - */ - bhm.xSpacerSize = function (_) { - return arguments.length ? (xSpacerSize = _, bhm) : xSpacerSize; - }; - /** - * Gets / sets the ySize - * (see {@link bubbleHeatmap#ySize}) - * @param {number} [_=none] - * @returns {bubbleHeatmap | number} - * @memberof bubbleHeatmap - * @property - * by default ySize = undefined - */ - bhm.ySize = function (_) { - return arguments.length ? (ySize = _, bhm) : ySize; - }; - /** - * Gets / sets the ySpacerSize - * (see {@link bubbleHeatmap#ySpacerSize}) - * @param {number} [_=none] - * @returns {bubbleHeatmap | number} - * @memberof bubbleHeatmap - * @property - * by default ySpacerSize = undefined - */ - bhm.ySpacerSize = function (_) { - return arguments.length ? (ySpacerSize = _, bhm) : ySpacerSize; - }; - // bhm.yKeySortingFunction = function(_) { return arguments.length ? (yKeySortingFunction = _, bhm) : yKeySortingFunction; } - // bhm.xKeySortingFunction = function(_) { return arguments.length ? (xKeySortingFunction = _, bhm) : xKeySortingFunction; } - - - function bhm() { - var bgcpRect = { x: 0, y: 0, width: spaceX, height: spaceY }; - var container = setupContainer(selection, namespace, bgcpRect, backgroundFill); - - cellKeys = d3.keys(data); - cellKeys.sort(function (a, b) { - return xKeySortingFunction(a, b) || yKeySortingFunction(a, b); - }); - log('bubbleHeatmap', 'cells are sorted by', cellKeys); - - xValues = unique(cellKeys.map(xExtractor)); - yValues = unique(cellKeys.map(yExtractor)); - rValues = unique(cellKeys.map(rExtractor)); - vValues = unique(cellKeys.map(vExtractor)); - log('bubbleHeatmap', 'x and y keys are', { x: xValues, y: yValues }); - - var xDim = xValues.length, - yDim = yValues.length; - - var extent = [Math.min.apply(Math, toConsumableArray(rValues)) - domainPadding, Math.max.apply(Math, toConsumableArray(rValues)) + domainPadding]; - - ySize = calculateWidthOfObject(spaceY, yDim, minObjectSize, maxObjectSize, objectSpacer, overflowQ); - xSize = calculateWidthOfObject(spaceX, xDim, minObjectSize, maxObjectSize, objectSpacer, overflowQ); - ySpacerSize = calculateWidthOfSpacer(yValues, spaceY, ySize, yDim, objectSpacer, overflowQ); - xSpacerSize = calculateWidthOfSpacer(xValues, spaceX, xSize, xDim, objectSpacer, overflowQ); - log('bubbleHeatmap', 'size of', { x: xSize, y: ySize }); - - scale.domain(extent).range([Math.min(minObjectSize / 2, Math.min(ySize, xSize) / 2), Math.min(ySize, xSize) / 2]); - - var ySpacer = groupingSpacer().horizontalQ(false).moveby('category').numberOfObjects(yDim).objectClass(hypenate(objectClass, 'row')).objectSize(ySize).spacerSize(ySpacerSize).transitionDuration(transitionDuration).easeFunc(easeFunc).namespace('row'); - - var xSpacer = groupingSpacer().horizontalQ(true).moveby('category').numberOfObjects(xDim).objectClass(objectClass).objectSize(xSize).spacerSize(xSpacerSize).transitionDuration(transitionDuration).easeFunc(easeFunc); - - ySpacer(container, yValues, 0); - container.selectAll('g.' + hypenate(objectClass, 'row')).each(function (d, i) { - xSpacer(d3.select(this), xValues, 0); - }); - var cells = container.selectAll('g:not(.to-remove).' + objectClass).data(cellKeys); - - var parentIndexArray = []; - cells.each(function (d, i) { - parentIndexArray.push(Number(d3.select(this).attr('parent-index'))); - }); - - colorFunction$$1 = colorFunction$$1.colorBy() == 'index' ? colorFunction$$1.dataExtent([0, Math.max.apply(Math, parentIndexArray)]) : colorFunction$$1.dataExtent([0, Math.max.apply(Math, toConsumableArray(vValues))]); - - cells.each(function (key, i) { - log('bubbleHeatmap', 'each cell', { key: key, index: i, node: d3.select(this).node() }); - - var t = d3.select(this), - currentData = data[key], - value = vExtractor(key, i), - radius = rExtractor(key, i), - i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'), - fillColor = colorFunction$$1(key, value, i, 'fill'), - // prevent duplicate computation - strokeColor = colorFunction$$1(key, value, i, 'stroke'); - - log('bubbleHeatmap', 'radius', { radius: radius, scaled: scale(radius), extent: extent, range: scale.range() }); - - var c = safeSelect(t, 'circle', hypenate(objectClass, 'circle')); - c.attr('cx', xSize / 2).attr('cy', ySize / 2).attr('r', scale(radius)).attr('fill', fillColor).attr('stroke', strokeColor).attr('stroke-width', bubbleStrokeWidth); - }); - - tooltip$$1.selection(cells.selectAll('circle.' + hypenate(objectClass, 'circle'))).data(data); - // .keys(['r', 'v']) - // .header(function(d, i){return hypenate(data[d][xKey], data[d][yKey]) }) - - tooltip$$1(); - } - - return bhm; - } - - /** - * Creates a heatmap - * - * {@link https://sumneuron.gitlab.io/d3sm/demos/heatmap-heatmap/index.html Demo} - * @constructor heatmap - * @param {d3.selection} selection - * @namespace heatmap - * @returns {function} heatmap - */ - function heatmap(selection) { - var - /** - * Data to plot. Assumed to be a object, where each key corresponds to a cell - * (see {@link heatmap#data}) - * @param {Object} [data=undefined] - * @memberof heatmap# - * @property - */ - data, - - - /** - * Amount of horizontal space (in pixels) avaible to render the heatmap in - * (see {@link heatmap#spaceX}) - * @param {number} [spaceX=undefined] - * @memberof heatmap# - * @property - */ - spaceX, - - /** - * Amount of vertical space (in pixels) avaible to render the heatmap in - * (see {@link heatmap.spaceY}) - * @param {number} [spaceY=undefined] - * @memberof heatmap# - * @property - */ - spaceY, - - - /** - * The internal key of the cell specifiying to which x axis key it belongs - * (see {@link heatmap.xKey}) - * @param {string} [xKey='x'] - * @memberof heatmap# - * @property - */ - xKey = 'x', - - /** - * The internal key of the cell specifiying to which y axis key it belongs - * (see {@link heatmap.yKey}) - * @param {string} [yKey='y'] - * @memberof heatmap# - * @property - */ - yKey = 'y', - - - /** - * The internal key of the cell specifiying what value to use to determine the color - * (see {@link heatmap.vKey}) - * @param {string} [vKey='v'] - * @memberof heatmap# - * @property - */ - vKey = 'v', - - - /** - * Function for extracting the the value from xKey. - * (see {@link heatmap.xExtractor}) - * @param {function} [xExtractor=function(key, i) { return data[key][xKey] }] - * @returns {string} - * @memberof heatmap# - * @property - */ - xExtractor = function xExtractor(key, i) { - return data[key][xKey]; - }, - - /** - * Function for extracting the the value from yKey. - * (see {@link heatmap.yExtractor}) - * @param {function} [yExtractor=function(key, i) { return data[key][yKey] }] - * @returns {string} - * @memberof heatmap# - * @property - */ - yExtractor = function yExtractor(key, i) { - return data[key][yKey]; - }, - - - /** - * Function for extracting the the value from vKey. - * (see {@link heatmap.vExtractor}) - * @param {function} [vExtractor=function(key, i) { return data[key][vKey] }] - * @returns {number} - * @memberof heatmap# - * @property - */ - vExtractor = function vExtractor(key, i) { - return data[key][vKey]; - }, - - - /** - * Whether or not to allow heatmap to render elements pass the main spatial dimension - * given the orientation (see {@link heatmap#orient}), where {@link heatmap#orient}="bottom" or {@link heatmap#orient}="top" - * the main dimension is {@link heatmap#spaceX} and where {@link heatmap#orient}="left" or {@link heatmap#orient}="right" - * the main dimension is {@link heatmap#spaceY} - * @param {boolean} [overflowQ=false] - * @memberof heatmap# - * @property - */ - overflowQ = false, - - - /** - * Default space for the spacer (percentage) of main dimension given the orientation - * (see {@link heatmap#orient}), where {@link heatmap#orient}="horizontal" - * the main dimension is {@link heatmap#spaceX} and where {@link heatmap#orient}="vertical" - * the main dimension is {@link heatmap#spaceY} between bubbles - * @param {number} [objectSpacer=0.0] - * @memberof heatmap# - * @property - */ - objectSpacer = 0.0, - - /** - * The minimum size that an object can be in the y dimension - * @param {number} [minObjectSize=50] - * @memberof heatmap# - * @property - */ - yMinObjectSize = 50, - - /** - * The minimum size that an object can be in the x dimension - * @param {number} [minObjectSize=50] - * @memberof heatmap# - * @property - */ - xMinObjectSize = 50, - - /** - * The maximum size that an object can be in the x dimension - * @param {number} [maxObjectSize=100] - * @memberof heatmap# - * @property - */ - xMaxObjectSize = 100, - - /** - * The maximum size that an object can be in the y dimension - * @param {number} [maxObjectSize=100] - * @memberof heatmap# - * @property - */ - yMaxObjectSize = 100, - - - /** - * The stroke width of the bubbles - * @param {number} [objectStrokeWidth=2] - * @memberof heatmap# - * @property - */ - objectStrokeWidth = 2, - - // colorFunc = colorFunction(), - - /** - * Color of the background - * @param {string} [backgroundFill="transparent"] - * @memberof heatmap# - * @property - */ - backgroundFill = 'transparent', - - /** - * Namespace for all items made by this instance of heatmap - * @param {string} [namespace="d3sm-heatmap"] - * @memberof heatmap# - * @property - */ - namespace = 'd3sm-heatmap', - - /** - * Class name for heatmap container ( element) - * @param {string} [objectClass="heatmap"] - * @memberof heatmap# - * @property - */ - objectClass = 'heatmap', - - /** - * Duration of all transitions of this element - * @param {number} [transitionDuration=1000] - * @memberof heatmap# - * @property - */ - transitionDuration = 1000, - - /** - * Easing function for transitions - * @param {d3.ease} [easeFunc=d3.easeExp] - * @memberof heatmap# - * @property - */ - easeFunc = d3.easeExp, - - - /** - * Stores the keys of all the cells - * Calculated after heatmap called. - * @param {string[]} [cellKeys=undefined] - * @memberof heatmap# - * @property - */ - cellKeys, - - /** - * Stores the list of unique xValues - * Calculated after heatmap called. - * @param {string[]} [xValues=undefined] - * @memberof heatmap# - * @property - */ - xValues, - - /** - * Stores the list of unique yValues - * Calculated after heatmap called. - * @param {string[]} [yValues=undefined] - * @memberof heatmap# - * @property - */ - yValues, - - /** - * Stores the list of unique vValues - * Calculated after heatmap called. - * @param {string[]} [vValues=undefined] - * @memberof heatmap# - * @property - */ - vValues, - xKeySortingFunction = function xKeySortingFunction(a, b) { - return xValues.indexOf(xExtractor(a)) - xValues.indexOf(xExtractor(b)); - }, - yKeySortingFunction = function yKeySortingFunction(a, b) { - return yValues.indexOf(yExtractor(a)) - yValues.indexOf(yExtractor(b)); - }, - /** - * Instance of ColorFunction with .colorBy set to 'category' - * @function colorFunction - * @memberof heatmap# - * @property - */ - colorFunction$$1 = colorFunction().colorBy('category'), - - /** - * Instance of Tooltip - * @function tooltip - * @memberof heatmap# - * @property - */ - tooltip$$1 = tooltip(), - - - /** - * store the size the heatmap could be in the x dimension - * the actuall size of the heatmap will be the min of xSize and {@link heatmap#ySize} - * Calculated after heatmap called. - * @param {string[]} [xSize=undefined] - * @memberof heatmap# - * @property - */ - xSize, - - /** - * store the size of the spacer in the x dimension - * Calculated after heatmap called. - * @param {string[]} [xSpacerSize=undefined] - * @memberof heatmap# - * @property - */ - xSpacerSize, - - - /** - * store the size the heatmap could be in the y dimension - * the actuall size of the heatmap will be the min of xSize and {@link heatmap#xSize} - * Calculated after heatmap called. - * @param {string[]} [ySize=undefined] - * @memberof heatmap# - * @property - */ - ySize, - - /** - * store the size of the spacer in the y dimension. - * Calculated after heatmap called. - * @param {string[]} [xSpacerSize=undefined] - * @memberof heatmap# - * @property - */ - ySpacerSize; - - /** - * Gets or sets the selection in which items are manipulated - * @param {d3.selection} [_=none] - * @returns {heatmap | d3.selection} - * @memberof heatmap - * @property - * by default selection = selection - */ - hm.selection = function (_) { - return arguments.length ? (selection = _, hm) : selection; - }; - /** - * Gets or sets the data - * (see {@link heatmap#data}) - * @param {number} [_=none] - * @returns {heatmap | object} - * @memberof heatmap - * @property - */ - hm.data = function (_) { - return arguments.length ? (data = _, hm) : data; - }; - // hm.orient = function(_) { return arguments.length ? (orient = _, hm) : orient; } - /** - * Gets or sets the amount of horizontal space in which items are manipulated - * (see {@link heatmap#spaceX}) - * @param {number} [_=none] should be a number > 0 - * @returns {heatmap | number} - * @memberof heatmap - * @property - * by default spaceX = undefined - */ - hm.spaceX = function (_) { - return arguments.length ? (spaceX = _, hm) : spaceX; - }; - /** - * Gets or sets the amount of vertical space in which items are manipulated - * (see {@link heatmap#spaceY}) - * @param {number} [_=none] should be a number > 0 - * @returns {heatmap | number} - * @memberof heatmap - * @property - * by default spaceY = undefined - */ - hm.spaceY = function (_) { - return arguments.length ? (spaceY = _, hm) : spaceY; - }; - - /** - * Gets or sets the xKey - * (see {@link heatmap#xKey}) - * @param {string} [_=none] - * @returns {heatmap | string} - * @memberof heatmap - * @property - * by default xKey = 'x' - */ - hm.xKey = function (_) { - return arguments.length ? (xKey = _, hm) : xKey; - }; - /** - * Gets or sets the yKey - * (see {@link heatmap#yKey}) - * @param {string} [_=none] - * @returns {heatmap | string} - * @memberof heatmap - * @property - * by default yKey = 'y' - */ - hm.yKey = function (_) { - return arguments.length ? (yKey = _, hm) : yKey; - }; - - /** - * Gets or sets the vKey - * (see {@link heatmap#vKey}) - * @param {string} [_=none] - * @returns {heatmap | string} - * @memberof heatmap - * @property - * by default vKey = 'y' - */ - hm.vKey = function (_) { - return arguments.length ? (vKey = _, hm) : vKey; - }; - - /** - * Gets or sets the cellKeys - * (see {@link heatmap#cellKeys}) - * @param {string[]} [_=none] - * @returns {heatmap | string[]} - * @memberof heatmap - * @property - * by default cellKeys = undefined - */ - hm.cellKeys = function (_) { - return arguments.length ? (cellKeys = _, hm) : cellKeys; - }; - /** - * Gets or sets the xValues - * (see {@link heatmap#xValues}) - * @param {string[]} [_=none] - * @returns {heatmap | string[]} - * @memberof heatmap - * @property - * by default xValues = undefined - */ - hm.xValues = function (_) { - return arguments.length ? (xValues = _, hm) : xValues; - }; - /** - * Gets or sets the yValues - * (see {@link heatmap#yValues}) - * @param {string[]} [_=none] - * @returns {heatmap | string[]} - * @memberof heatmap - * @property - * by default yValues = undefined - */ - hm.yValues = function (_) { - return arguments.length ? (yValues = _, hm) : yValues; - }; - /** - * Gets or sets the vValues - * (see {@link heatmap#vValues}) - * @param {number[]} [_=none] - * @returns {heatmap | number[]} - * @memberof heatmap - * @property - * by default vValues = undefined - */ - hm.vValues = function (_) { - return arguments.length ? (vValues = _, hm) : vValues; - }; - - /** - * Gets or sets the xExtractor - * (see {@link heatmap#xExtractor}) - * @param {function} [_=none] - * @returns {heatmap | function} - * @memberof heatmap - * @property - * by default xExtractor = undefined - */ - hm.xExtractor = function (_) { - return arguments.length ? (xExtractor = _, hm) : xExtractor; - }; - /** - * Gets or sets the yExtractor - * (see {@link heatmap#yExtractor}) - * @param {function} [_=none] - * @returns {heatmap | function} - * @memberof heatmap - * @property - * by default yExtractor = undefined - */ - hm.yExtractor = function (_) { - return arguments.length ? (yExtractor = _, hm) : yExtractor; - }; - /** - * Gets or sets the vExtractor - * (see {@link heatmap#vExtractor}) - * @param {function} [_=none] - * @returns {heatmap | function} - * @memberof heatmap - * @property - * by default vExtractor = undefined - */ - hm.vExtractor = function (_) { - return arguments.length ? (vExtractor = _, hm) : vExtractor; - }; - - /** - * Gets / sets whether or not heatmap is allowed to go beyond specified dimensions - * (see {@link heatmap#spaceX}) - * @param {boolean} [_=none] - * @returns {heatmap | boolean} - * @memberof heatmap - * @property - * by default overflowQ = false - */ - hm.overflowQ = function (_) { - return arguments.length ? (overflowQ = _, hm) : overflowQ; - }; - /** - * Gets / sets objectSpacer - * (see {@link heatmap#objectSpacer}) - * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap - * @property - * by default objectSpacer = 0.0 - */ - hm.objectSpacer = function (_) { - return arguments.length ? (objectSpacer = _, objectSpacer) : data; - }; - /** - * Gets / sets the minObjectSize - * (see {@link heatmap#minObjectSize}) - * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap - * @property - * by default minObjectSize = 50 - */ - hm.yMinObjectSize = function (_) { - return arguments.length ? (yMinObjectSize = _, hm) : yMinObjectSize; - }; - /** - * Gets / sets the maxObjectSize - * (see {@link heatmap#maxObjectSize}) - * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap - * @property - * by default maxObjectSize = 100 - */ - hm.yMaxObjectSize = function (_) { - return arguments.length ? (yMaxObjectSize = _, hm) : yMaxObjectSize; - }; - /** - * Gets / sets the minObjectSize - * (see {@link heatmap#minObjectSize}) - * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap - * @property - * by default minObjectSize = 50 - */ - hm.xMinObjectSize = function (_) { - return arguments.length ? (xMinObjectSize = _, hm) : xMinObjectSize; - }; - /** - * Gets / sets the maxObjectSize - * (see {@link heatmap#maxObjectSize}) - * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap - * @property - * by default maxObjectSize = 100 - */ - hm.xMaxObjectSize = function (_) { - return arguments.length ? (xMaxObjectSize = _, hm) : xMaxObjectSize; - }; - /** - * Gets / sets the objectStrokeWidth - * (see {@link heatmap#objectStrokeWidth}) - * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap - * @property - * by default objectStrokeWidth = 2 - */ - hm.objectStrokeWidth = function (_) { - return arguments.length ? (objectStrokeWidth = _, hm) : objectStrokeWidth; - }; - /** - * Gets / sets the backgroundFill - * (see {@link heatmap#backgroundFill}) - * @param {string} [_=none] - * @returns {heatmap | string} - * @memberof heatmap - * @property - * by default backgroundFill = 'transparent' - */ - hm.backgroundFill = function (_) { - return arguments.length ? (backgroundFill = _, hm) : backgroundFill; - }; - /** - * Gets / sets the namespace - * (see {@link heatmap#namespace}) - * @param {string} [_=none] - * @returns {heatmap | string} - * @memberof heatmap - * @property - * by default namespace = 'd3sm-heatmap' - */ - hm.namespace = function (_) { - return arguments.length ? (namespace = _, hm) : namespace; - }; - /** - * Gets / sets the objectClass - * (see {@link heatmap#objectClass}) - * @param {string} [_=none] - * @returns {heatmap | string} - * @memberof heatmap - * @property - * by default objectClass = 'tick-group' - */ - hm.objectClass = function (_) { - return arguments.length ? (objectClass = _, hm) : objectClass; - }; - /** - * Gets / sets the transitionDuration - * (see {@link heatmap#transitionDuration}) - * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap - * @property - * by default transitionDuration = 1000 - */ - hm.transitionDuration = function (_) { - return arguments.length ? (transitionDuration = _, hm) : transitionDuration; - }; - /** - * Gets / sets the easeFunc - * (see {@link heatmap#easeFunc}) - * @param {d3.ease} [_=none] - * @returns {heatmap | d3.ease} - * @memberof heatmap - * @property - * by default easeFunc = d3.easeExp - */ - hm.easeFunc = function (_) { - return arguments.length ? (easeFunc = _, hm) : easeFunc; - }; - - /** - * Gets / sets the tooltip - * (see {@link heatmap#tooltip}) - * @param {tooltip} [_=none] - * @returns {heatmap | tooltip} - * @memberof heatmap - * @property - * by default tooltip = tooltip() - */ - hm.tooltip = function (_) { - return arguments.length ? (tooltip$$1 = _, hm) : tooltip$$1; - }; - - /** - * Gets / sets the colorFunction - * (see {@link heatmap#colorFunction}) - * @param {colorFunction} [_=none] - * @returns {heatmap | colorFunction} - * @memberof heatmap - * @property - * by default colorFunction = colorFunction() - */ - hm.colorFunction = function (_) { - return arguments.length ? (colorFunction$$1 = _, hm) : colorFunction$$1; - }; - - /** - * Gets / sets the xSize - * (see {@link heatmap#xSize}) - * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap - * @property - * by default xSize = undefined - */ - hm.xSize = function (_) { - return arguments.length ? (xSize = _, hm) : xSize; - }; - /** - * Gets / sets the xSpacerSize - * (see {@link heatmap#xSpacerSize}) - * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap - * @property - * by default xSpacerSize = undefined - */ - hm.xSpacerSize = function (_) { - return arguments.length ? (xSpacerSize = _, hm) : xSpacerSize; - }; - /** - * Gets / sets the ySize - * (see {@link heatmap#ySize}) - * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap - * @property - * by default ySize = undefined - */ - hm.ySize = function (_) { - return arguments.length ? (ySize = _, hm) : ySize; - }; - /** - * Gets / sets the ySpacerSize - * (see {@link heatmap#ySpacerSize}) - * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap - * @property - * by default ySpacerSize = undefined - */ - hm.ySpacerSize = function (_) { - return arguments.length ? (ySpacerSize = _, hm) : ySpacerSize; - }; - // hm.yKeySortingFunction = function(_) { return arguments.length ? (yKeySortingFunction = _, hm) : yKeySortingFunction; } - // hm.xKeySortingFunction = function(_) { return arguments.length ? (xKeySortingFunction = _, hm) : xKeySortingFunction; } - - - function hm() { - var bgcpRect = { x: 0, y: 0, width: spaceX, height: spaceY }; - var container = setupContainer(selection, namespace, bgcpRect, backgroundFill); - - cellKeys = d3.keys(data); - - xValues = unique(cellKeys.map(xExtractor)); - yValues = unique(cellKeys.map(yExtractor)); - vValues = unique(cellKeys.map(vExtractor)); - - cellKeys.sort(function (a, b) { - return xKeySortingFunction(a, b) || yKeySortingFunction(a, b); - }); - log('heatmap', 'cells are sorted by', cellKeys); - - log('heatmap', 'x and y keys are', { x: xValues, y: yValues }); - - var xDim = xValues.length, - yDim = yValues.length; - - ySize = calculateWidthOfObject(spaceY, yDim, yMinObjectSize, yMaxObjectSize, objectSpacer, overflowQ); - xSize = calculateWidthOfObject(spaceX, xDim, xMinObjectSize, xMaxObjectSize, objectSpacer, overflowQ); - ySpacerSize = calculateWidthOfSpacer(yValues, spaceY, ySize, yDim, objectSpacer, overflowQ); - xSpacerSize = calculateWidthOfSpacer(xValues, spaceX, xSize, xDim, objectSpacer, overflowQ); - // console.table({ - // x:{ - // object: xSize, - // spacer: xSpacerSize, - // dim: xDim - // }, - // y:{ - // object: ySize, - // spacer: ySpacerSize, - // dim: yDim - // } - // - // }) - log('heatmap', 'size of', { x: xSize, y: ySize }); - - var ySpacer = groupingSpacer().horizontalQ(false).moveby('category').numberOfObjects(yDim).objectClass(hypenate(objectClass, 'row')).objectSize(ySize + ySpacerSize).spacerSize(0).transitionDuration(transitionDuration).easeFunc(easeFunc).namespace('row'); - - var xSpacer = groupingSpacer().horizontalQ(true).moveby('category').numberOfObjects(xDim).objectClass(objectClass).objectSize(xSize + xSpacerSize).spacerSize(0).transitionDuration(transitionDuration).easeFunc(easeFunc); - - ySpacer(container, yValues, 0); - container.selectAll('g.' + hypenate(objectClass, 'row')).each(function (d, i) { - xSpacer(d3.select(this), xValues, 0); - }); - - var cells = container.selectAll('g:not(.to-remove).' + objectClass); - - if (cellKeys.length != yValues.length * xValues.length) { - var lookup = {}; - cellKeys.map(function (k, i) { - lookup[xExtractor(k) + '::' + yExtractor(k)] = k; - }); - - var positionedCellKeys = []; - for (var i = 0; i < yValues.length; i++) { - for (var j = 0; j < xValues.length; j++) { - var lookupValue = lookup[xValues[j] + "::" + yValues[i]]; - if (lookupValue == undefined) { - positionedCellKeys.push(undefined); - } else { - positionedCellKeys.push(lookupValue); - } - } - } - - cells.data(positionedCellKeys); - - // maybe breaks this - // !!!!!! IMPORTANT NOTE TODO LOOK HERE BUG - cellKeys = positionedCellKeys; - } else { - cells.data(cellKeys); - } - - var parentIndexArray = []; - cells.each(function (d, i) { - parentIndexArray.push(Number(d3.select(this).attr('parent-index'))); - }); - - colorFunction$$1 = colorFunction$$1.colorBy() == 'index' ? colorFunction$$1.dataExtent([0, Math.max.apply(Math, parentIndexArray)]) : colorFunction$$1.dataExtent([0, Math.max.apply(Math, toConsumableArray(vValues))]); - - cells.each(function (key, i) { - log('heatmap', 'each cell', { key: key, index: i, node: d3.select(this).node() }); - - var t = d3.select(this); - if (key == undefined) { - return; - } - var currentData = data[key], - value = vExtractor(key, i), - i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'), - fillColor = colorFunction$$1(key, value, i, 'fill'), - // prevent duplicate computation - strokeColor = colorFunction$$1(key, value, i, 'stroke'); - - var c = safeSelect(t, 'rect', hypenate(objectClass, 'rect')); - c.attr('width', xSize + xSpacerSize - objectStrokeWidth).attr('height', ySize + ySpacerSize - objectStrokeWidth).attr('fill', fillColor).attr('x', objectStrokeWidth / 2).attr('y', objectStrokeWidth / 2).attr('stroke', "#000").attr('stroke-width', objectStrokeWidth); - }); - - tooltip$$1.selection(cells.selectAll('rect.' + hypenate(objectClass, 'rect'))).data(data); - // .keys(['r', 'v']) - // .header(function(d, i){return hypenate(data[d][xKey], data[d][yKey]) }) - - tooltip$$1(); - } - - return hm; - } - - /******************************************************************************* - ** ** - ** ** - ** BOX AND WHISKER ** - ** ** - ** ** - *******************************************************************************/ - /** - * Creates a boxwhisker - * - * {@link https://sumneuron.gitlab.io/d3sm/demos/box-whiskers/index.html Demo} - * @constructor boxwhisker - * @param {d3.selection} selection - * @namespace boxwhisker - * @returns {function} boxwhisker - */ - function boxwhisker(selection) { - var - /** - * Data to plot. Assumed to be a object, where each key corresponds to a box - * (see {@link boxwhisker#data}) - * @param {Object} [data=undefined] - * @memberof boxwhisker# - * @property - */ - data, - - /** - * Which direction to render the boxes in - * (see {@link boxwhisker#orient}) - * @param {number} [orient='horizontal'] - * @memberof boxwhisker# - * @property - */ - orient = 'horizontal', - - /** - * Amount of horizontal space (in pixels) avaible to render the boxwhisker in - * (see {@link boxwhisker#spaceX}) - * @param {number} [spaceX=undefined] - * @memberof boxwhisker# - * @property - */ - spaceX, - - /** - * Amount of vertical space (in pixels) avaible to render the boxwhisker in - * (see {@link boxwhisker.spaceY}) - * @param {number} [spaceY=undefined] - * @memberof boxwhisker# - * @property - */ - spaceY, - - /** - * Whether or not to allow boxwhisker to render elements pass the main spatial dimension - * given the orientation (see {@link boxwhisker#orient}), where {@link boxwhisker#orient}="horizontal" - * the main dimension is {@link boxwhisker#spaceX} and where {@link boxwhisker#orient}="vertical" - * the main dimension is {@link boxwhisker#spaceY} - * @param {boolean} [overflowQ=false] - * @memberof boxwhisker# - * @property - */ - overflowQ = true, - - - /** - * An array - putatively of other arrays - depicting how boxes should be arranged - * @param {Array[]} [grouping=undefined] - * @memberof boxwhisker# - * @property - */ - grouping, - quartilesKey = 'quartiles', - // key in object where quartiles are stored - quartilesKeys = ["0.00", "0.25", "0.50", "0.75", "1.00"], - // keys in quartiles object mapping values to min, q1, q2, q3 and max - - - /** - * How to get the value of the boxwhisker - * @param {function} [valueExtractor=function(key, index) { return data[key][quartilesKey] }] - * @memberof boxwhisker# - * @property - */ - valueExtractor = function valueExtractor(key, index) { - return data[key][quartilesKey]; - }, - - /** - * How to sort the boxes - if {@link boxwhisker#grouping} is not provided. - * @param {function} [sortingFunction=descending] - * @memberof boxwhisker# - * @property - * default - * function(keyA, keyB) {return d3.descending( - * valueExtractor(keyA)[quartilesKeys[4]], - * valueExtractor(keyB)[quartilesKeys[4]] - * )} - */ - sortingFunction = function sortingFunction(keyA, keyB) { - return d3.descending(valueExtractor(keyA)[quartilesKeys[4]], valueExtractor(keyB)[quartilesKeys[4]]); - }, - - /** - * The scale for which boxwhisker values should be transformed by - * @param {d3.scale} [scale=d3.scaleLinear] - * @memberof boxwhisker# - * @property - */ - scale = d3.scaleLinear(), - - /** - * The padding for the domain of the scale (see {@link boxwhisker#scale}) - * @param {number} [domainPadding=0.5] - * @memberof boxwhisker# - * @property - */ - domainPadding = 0.5, - - /** - * Default space for the spacer (percentage) of main dimension given the orientation - * (see {@link boxwhisker#orient}), where {@link boxwhisker#orient}="horizontal" - * the main dimension is {@link boxwhisker#spaceX} and where {@link boxwhisker#orient}="vertical" - * the main dimension is {@link boxwhisker#spaceY} between boxes - * @param {number} [objectSpacer=0.05] - * @memberof boxwhisker# - * @property - */ - objectSpacer = 0.05, - - /** - * The minimum size that an object can be - * @param {number} [minObjectSize=15] - * @memberof boxwhisker# - * @property - */ - minObjectSize = 15, - - /** - * The maximum size that an object can be - * @param {number} [maxObjectSize=50] - * @memberof boxwhisker# - * @property - */ - maxObjectSize = 50, - - /** - * Percent of box and whisker size that whiskers will be rendered - * see {@link boxwhisker#objectSize} - * @param {number} [whiskerWidthPercent=0.6] - * @memberof boxwhisker# - * @property - */ - whiskerWidthPercent = .6, - - /** - * Instance of ColorFunction - * @param {function} [colorFunction = colorFunction()] - * @memberof boxwhisker# - * @property - */ - colorFunction$$1 = colorFunction(), - - /** - * The stroke width of the boxes - * @param {number} [boxStrokeWidth=2] - * @memberof boxwhisker# - * @property - */ - boxStrokeWidth = 2, - - /** - * The stroke width of the whiskers - * @param {number} [whiskerStrokeWidth=2] - * @memberof boxwhisker# - * @property - */ - whiskerStrokeWidth = 2, - - - /** - * Color of the background - * @param {string} [backgroundFill="transparent"] - * @memberof boxwhisker# - * @property - */ - backgroundFill = 'transparent', - - /** - * Namespace for all items made by this instance of boxwhisker - * @param {string} [namespace="d3sm-box-whisker"] - * @memberof boxwhisker# - * @property - */ - namespace = 'd3sm-box-whisker', - - /** - * Class name for boxwhisker container ( element) - * @param {string} [objectClass="box-whisk"] - * @memberof boxwhisker# - * @property - */ - objectClass = 'box-whisk', - - - /** - * Duration of all transitions of this element - * @param {number} [transitionDuration=1000] - * @memberof boxwhisker# - * @property - */ - transitionDuration = 1000, - - /** - * Easing function for transitions - * @param {d3.ease} [easeFunc=d3.easeExp] - * @memberof boxwhisker# - * @property - */ - easeFunc = d3.easeExp, - - - /** - * The keys of the boxes - * @param {string[]} [boxKeys=undefined] - * @memberof boxwhisker# - * @property - */ - boxKeys, - - /** - * The values of the boxes - * @param {string[]} [boxValues=undefined] - * @memberof boxwhisker# - * @property - */ - boxValues, - - /** - * The objectSize (actual width) used by the boxes - * @param {number} [objectSize=undefined] - * @memberof boxwhisker# - * @property - */ - objectSize, - - /** - * The spacerSize (actual width) used by the spacers between the boxes - * @param {number} [spacerSize=undefined] - * @memberof boxwhisker# - * @property - */ - spacerSize, - - /** - * Instance of Tooltip - * @param {function} [tooltip=tooltip()] - * @memberof boxwhisker# - * @property - */ - tooltip$$1 = tooltip(); - /** - * Gets or sets the selection in which items are manipulated - * @param {d3.selection} [_=none] - * @returns {boxwhisker | d3.selection} - * @memberof boxwhisker - * @property - * by default selection = selection - */ - boxwhisker.selection = function (_) { - return arguments.length ? (selection = _, boxwhisker) : selection; - }; - /** - * Gets or sets the data - * (see {@link boxwhisker#data}) - * @param {number} [_=none] - * @returns {boxwhisker | object} - * @memberof boxwhisker - * @property - */ - boxwhisker.data = function (_) { - return arguments.length ? (data = _, boxwhisker) : data; - }; - /** - * Gets or sets the orient of the boxes - * (see {@link boxwhisker#orient}) - * @param {number} [_=none] - * @returns {boxwhisker | object} - * @memberof boxwhisker - * @property - */ - boxwhisker.orient = function (_) { - return arguments.length ? (orient = _, boxwhisker) : orient; - }; - /** - * Gets or sets the amount of horizontal space in which items are manipulated - * (see {@link boxwhisker#spaceX}) - * @param {number} [_=none] should be a number > 0 - * @returns {boxwhisker | number} - * @memberof boxwhisker - * @property - * by default spaceX = undefined - */ - boxwhisker.spaceX = function (_) { - return arguments.length ? (spaceX = _, boxwhisker) : spaceX; - }; - /** - * Gets or sets the amount of vertical space in which items are manipulated - * (see {@link boxwhisker#spaceY}) - * @param {number} [_=none] should be a number > 0 - * @returns {boxwhisker | number} - * @memberof boxwhisker - * @property - * by default spaceY = undefined - */ - boxwhisker.spaceY = function (_) { - return arguments.length ? (spaceY = _, boxwhisker) : spaceY; - }; - /** - * Gets / sets whether or not boxwhisker is allowed to go beyond specified dimensions - * (see {@link boxwhisker#overflowQ}) - * @param {boolean} [_=none] - * @returns {boxwhisker | boolean} - * @memberof boxwhisker - * @property - * by default overflowQ = false - */ - boxwhisker.overflowQ = function (_) { - return arguments.length ? (overflowQ = _, boxwhisker) : overflowQ; - }; - /** - * Gets / sets the grouping of the boxes - * (see {@link boxwhisker#grouping}) - * @param {Array[]} [_=none] - * @returns {boxwhisker | Array[]} - * @memberof boxwhisker - * @property - * by default grouping = undefined - */ - boxwhisker.grouping = function (_) { - return arguments.length ? (grouping = _, boxwhisker) : grouping; - }; - /** - * Gets / sets the quartilesKey - * (see {@link boxwhisker#quartilesKey}) - * @param {string} [_=none] - * @returns {boxwhisker | string} - * @memberof boxwhisker - * @property - * by default quartilesKey = quartiles - */ - boxwhisker.quartilesKey = function (_) { - return arguments.length ? (quartilesKey = _, boxwhisker) : quartilesKey; - }; - /** - * Gets / sets the quartilesKeys - * (see {@link boxwhisker#quartilesKeys}) - * @param {string[]} [_=none] - * @returns {boxwhisker | string[]} - * @memberof boxwhisker - * @property - * by default quartilesKeys = [Q0, Q1, Q2, Q3, Q4] - */ - boxwhisker.quartilesKeys = function (_) { - return arguments.length ? (quartilesKeys = _, boxwhisker) : quartilesKeys; - }; - /** - * Gets / sets the valueExtractor - * (see {@link boxwhisker#valueExtractor}) - * @param {function} [_=none] - * @returns {boxwhisker | function} - * @memberof boxwhisker - * @property - * by default valueExtractor = function(key, index) { return data[key][quartilesKey] }, - */ - - boxwhisker.valueExtractor = function (_) { - return arguments.length ? (valueExtractor = _, boxwhisker) : valueExtractor; - }; - /** - * Gets / sets the sortingFunction - * (see {@link boxwhisker#sortingFunction}) - * @param {function} [_=none] - * @returns {boxwhisker | function} - * @memberof boxwhisker - * @property - * by default sortingFunction = function(keyA, keyB) {return d3.descending( - * valueExtractor(keyA)[quartilesKeys[4]], - * valueExtractor(keyB)[quartilesKeys[4]] - * )}, - */ - boxwhisker.sortingFunction = function (_) { - return arguments.length ? (sortingFunction = _, boxwhisker) : sortingFunction; - }; - /** - * Gets / sets the scale for which the boxwhisker values should be transformed by - * (see {@link boxwhisker#scale}) - * @param {d3.scale} [_=none] - * @returns {boxwhisker | d3.scale} - * @memberof boxwhisker - * @property - * by default scale = d3.scaleLinear() - */ - boxwhisker.scale = function (_) { - return arguments.length ? (scale = _, boxwhisker) : scale; - }; - /** - * Gets / sets the padding for the domain of the scale - * (see {@link boxwhisker#domainPadding}) - * @param {number} [_=none] - * @returns {boxwhisker | number} - * @memberof boxwhisker - * @property - * by default domainPadding = 0.5 - */ - boxwhisker.domainPadding = function (_) { - return arguments.length ? (domainPadding = _, boxwhisker) : domainPadding; - }; - /** - * Gets / sets objectSpacer - * (see {@link boxwhisker#objectSpacer}) - * @param {number} [_=none] - * @returns {boxwhisker | number} - * @memberof boxwhisker - * @property - * by default objectSpacer = 0.05 - */ - boxwhisker.objectSpacer = function (_) { - return arguments.length ? (objectSpacer = _, boxwhisker) : objectSpacer; - }; - /** - * Gets / sets the minObjectSize - * (see {@link boxwhisker#minObjectSize}) - * @param {number} [_=none] - * @returns {boxwhisker | number} - * @memberof boxwhisker - * @property - * by default minObjectSize = 15 - */ - boxwhisker.minObjectSize = function (_) { - return arguments.length ? (minObjectSize = _, boxwhisker) : minObjectSize; - }; - /** - * Gets / sets the maxObjectSize - * (see {@link boxwhisker#maxObjectSize}) - * @param {number} [_=none] - * @returns {boxwhisker | number} - * @memberof boxwhisker - * @property - * by default maxObjectSize = 50 - */ - boxwhisker.maxObjectSize = function (_) { - return arguments.length ? (maxObjectSize = _, boxwhisker) : maxObjectSize; - }; - /** - * Gets / sets the whiskerWidthPercent - * (see {@link boxwhisker#whiskerWidthPercent}) - * @param {number} [_=none] - * @returns {boxwhisker | number} - * @memberof boxwhisker - * @property - * by default maxObjectSize = 0.6 - */ - boxwhisker.whiskerWidthPercent = function (_) { - return arguments.length ? (whiskerWidthPercent = _, boxwhisker) : whiskerWidthPercent; - }; - /** - * Gets / sets the colorFunction - * (see {@link boxwhisker#colorFunction}) - * @param {colorFunction} [_=none] - * @returns {boxwhisker | colorFunction} - * @memberof boxwhisker - * @property - * by default colorFunction = colorFunction() - */ - boxwhisker.colorFunction = function (_) { - return arguments.length ? (colorFunction$$1 = _, boxwhisker) : colorFunction$$1; - }; - /** - * Gets / sets the boxStrokeWidth - * (see {@link boxwhisker#boxStrokeWidth}) - * @param {number} [_=none] - * @returns {boxwhisker | number} - * @memberof boxwhisker - * @property - * by default boxStrokeWidth = 2 - */ - boxwhisker.boxStrokeWidth = function (_) { - return arguments.length ? (boxStrokeWidth = _, boxwhisker) : boxStrokeWidth; - }; - /** - * Gets / sets the whiskerStrokeWidth - * (see {@link boxwhisker#whiskerStrokeWidth}) - * @param {number} [_=none] - * @returns {boxwhisker | number} - * @memberof boxwhisker - * @property - * by default whiskerStrokeWidth = 2 - */ - boxwhisker.whiskerStrokeWidth = function (_) { - return arguments.length ? (whiskerStrokeWidth = _, boxwhisker) : whiskerStrokeWidth; - }; - - /** - * Gets / sets the backgroundFill - * (see {@link boxwhisker#backgroundFill}) - * @param {string} [_=none] - * @returns {boxwhisker | string} - * @memberof boxwhisker - * @property - * by default backgroundFill = 'transparent' - */ - boxwhisker.backgroundFill = function (_) { - return arguments.length ? (backgroundFill = _, boxwhisker) : backgroundFill; - }; - /** - * Gets / sets the namespace - * (see {@link boxwhisker#namespace}) - * @param {string} [_=none] - * @returns {boxwhisker | string} - * @memberof boxwhisker - * @property - * by default namespace = 'd3sm-boxwhisker' - */ - boxwhisker.namespace = function (_) { - return arguments.length ? (namespace = _, boxwhisker) : namespace; - }; - /** - * Gets / sets the objectClass - * (see {@link boxwhisker#objectClass}) - * @param {string} [_=none] - * @returns {boxwhisker | string} - * @memberof boxwhisker - * @property - * by default objectClass = 'tick-group' - */ - boxwhisker.objectClass = function (_) { - return arguments.length ? (objectClass = _, boxwhisker) : objectClass; - }; - /** - * Gets / sets the transitionDuration - * (see {@link boxwhisker#transitionDuration}) - * @param {number} [_=none] - * @returns {boxwhisker | number} - * @memberof boxwhisker - * @property - * by default transitionDuration = 1000 - */ - boxwhisker.transitionDuration = function (_) { - return arguments.length ? (transitionDuration = _, boxwhisker) : transitionDuration; - }; - /** - * Gets / sets the easeFunc - * (see {@link boxwhisker#easeFunc}) - * @param {d3.ease} [_=none] - * @returns {boxwhisker | d3.ease} - * @memberof boxwhisker - * @property - * by default easeFunc = d3.easeExp - */ - boxwhisker.easeFunc = function (_) { - return arguments.length ? (easeFunc = _, boxwhisker) : easeFunc; - }; - - /** - * Gets / sets the barKeys - * (see {@link boxwhisker#boxKeys}) - * @param {string[]} [_=none] - * @returns {boxwhisker | string[]} - * @memberof boxwhisker - * @property - * by default boxKeys = undefined - */ - boxwhisker.boxKeys = function (_) { - return arguments.length ? (boxKeys = _, boxwhisker) : boxKeys; - }; - /** - * Gets / sets the boxValues - * (see {@link boxwhisker#boxValues}) - * @param {number[]} [_=none] - * @returns {boxwhisker | number[]} - * @memberof boxwhisker - * @property - * by default boxValues = undefined - */ - boxwhisker.boxValues = function (_) { - return arguments.length ? (boxValues = _, boxwhisker) : boxValues; - }; - /** - * Gets / sets the objectSize - * (see {@link boxwhisker#objectSize}) - * @param {number} [_=none] - * @returns {boxwhisker | number} - * @memberof boxwhisker - * @property - * by default objectSize = undefined - */ - boxwhisker.objectSize = function (_) { - return arguments.length ? (objectSize = _, boxwhisker) : objectSize; - }; - /** - * Gets / sets the spacerSize - * (see {@link boxwhisker#spacerSize}) - * @param {number} [_=none] - * @returns {boxwhisker | number} - * @memberof boxwhisker - * @property - * by default spacerSize = undefined - */ - boxwhisker.spacerSize = function (_) { - return arguments.length ? (spacerSize = _, boxwhisker) : spacerSize; - }; - /** - * Gets / sets the tooltip - * (see {@link boxwhisker#tooltip}) - * @param {tooltip} [_=none] - * @returns {boxwhisker | tooltip} - * @memberof boxwhisker - * @property - * by default tooltip = tooltip() - */ - boxwhisker.tooltip = function (_) { - return arguments.length ? (tooltip$$1 = _, boxwhisker) : tooltip$$1; - }; - - function boxwhisker() { - // for convenience in handling orientation specific values - var horizontalQ = orient == 'horizontal' ? true : false; - var verticalQ = !horizontalQ; - - // background cliping rectangle - var bgcpRect = { x: 0, y: 0, width: spaceX, height: spaceY }; - var container = setupContainer(selection, namespace, bgcpRect, backgroundFill); - - // if grouping is undefined sort keys by sorting funct - var ordered = grouping == undefined ? d3.keys(data).sort(sortingFunction) : grouping; - // to prevent re-calculation and getters to be passed to axes - boxKeys = flatten(ordered); - boxValues = boxKeys.map(valueExtractor); - - var numberOfObjects = boxKeys.length; - var extent = [Math.min.apply(Math, toConsumableArray(boxValues.map(function (d, i) { - return d[quartilesKeys[0]]; - }))) - domainPadding, Math.max.apply(Math, toConsumableArray(boxValues.map(function (d, i) { - return d[quartilesKeys[4]]; - }))) + domainPadding]; - - // set the scale - scale.domain(extent).range(horizontalQ ? [0, spaceY] : [spaceX, 0]); - var space = horizontalQ ? spaceX : spaceY; - // calculate object size - objectSize = calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ); - // calculate spacer size if needed - spacerSize = calculateWidthOfSpacer(boxKeys, space, objectSize, numberOfObjects, objectSpacer, overflowQ); - // make the nested groups - var spacerFunction = groupingSpacer().horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects).objectClass(objectClass).objectSize(objectSize).spacerSize(spacerSize).transitionDuration(transitionDuration).easeFunc(easeFunc).namespace(namespace); - - // move stuff - spacerFunction(container, ordered, 0); - - var parentIndexArray = []; - container.selectAll('g:not(.to-remove).' + objectClass).each(function (d, i) { - if (hasQ(boxKeys, d)) { - parentIndexArray.push(Number(d3.select(this).attr('parent-index'))); - } - }); - - colorFunction$$1 = colorFunction$$1.colorBy() == 'index' ? colorFunction$$1.dataExtent([0, Math.max.apply(Math, parentIndexArray)]) : colorFunction$$1.dataExtent(extent); - - // set attributes for box and whiskers - container.selectAll('g:not(.to-remove).' + objectClass).each(function (key, i) { - var t = d3.select(this), - currentData = data[key], - quartiles$$1 = valueExtractor(key, i), - q0 = quartiles$$1[quartilesKeys[0]], - q1 = quartiles$$1[quartilesKeys[1]], - q2 = quartiles$$1[quartilesKeys[2]], - q3 = quartiles$$1[quartilesKeys[3]], - q4 = quartiles$$1[quartilesKeys[4]]; - - var i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'), - fillColor = colorFunction$$1(key, q2, i, 'fill'), - // prevent duplicate computation - strokeColor = colorFunction$$1(key, q2, i, 'stroke'); - - var whisk = safeSelect(t, 'g', 'whisker'), - uWhisk = safeSelect(whisk, 'path', 'upper'), - lWhisk = safeSelect(whisk, 'path', 'lower'), - quart = safeSelect(t, 'g', 'quartile'), - uQuart = safeSelect(quart, 'rect', 'upper'), - lQuart = safeSelect(quart, 'rect', 'lower'), - mQuart = safeSelect(quart, 'circle', 'median'); - - // set upper quartile (q3) - uQuart.transition().duration(transitionDuration).ease(easeFunc).attr('width', horizontalQ ? objectSize : scale(q3) - scale(q2)).attr('height', verticalQ ? objectSize : scale(q3) - scale(q2)).attr('fill', fillColor).attr('stroke', strokeColor).attr('stroke-width', boxStrokeWidth).attr('transform', function (d, i) { - var x = horizontalQ ? 0 : scale(q2), - y = verticalQ ? 0 : scale(extent[1]) - scale(q3), - t = 'translate(' + x + ',' + y + ')'; - return t; - }); - - // set lower quartile (q1) - lQuart.transition().duration(transitionDuration).ease(easeFunc).attr('width', horizontalQ ? objectSize : scale(q2) - scale(q1)).attr('height', verticalQ ? objectSize : scale(q2) - scale(q1)).attr('fill', fillColor).attr('stroke', strokeColor).attr('stroke-width', boxStrokeWidth).attr('transform', function (d, i) { - var x = horizontalQ ? 0 : scale(q1), - y = verticalQ ? 0 : scale(extent[1]) - scale(q2), - t = 'translate(' + x + ',' + y + ')'; - return t; - }); - - // set median (q2) - mQuart.transition().duration(transitionDuration).ease(easeFunc).attr('r', function (d, i) { - var r = objectSize / 2; - var dif = (scale(q3) - scale(q1)) / 2; - return r > dif ? dif : r; - }).attr('fill', fillColor).attr('stroke', strokeColor).attr('stroke-width', boxStrokeWidth).attr('transform', function (d, i) { - var x = horizontalQ ? objectSize / 2 : scale(q2), - y = verticalQ ? objectSize / 2 : scale(extent[1]) - scale(q2), - t = 'translate(' + x + ',' + y + ')'; - return t; - }); - - // set lower whisker (min) - lWhisk.transition().duration(transitionDuration).ease(easeFunc).attr('d', function (dd, ii) { - var dir = false, - x = 0, - y = 0, - h = horizontalQ ? scale(q1) - scale(q0) : objectSize, - w = verticalQ ? scale(q1) - scale(q0) : objectSize; - return whiskerPath(dir, x, y, w, h, whiskerWidthPercent, orient); - }).attr('transform', function (d, i) { - var x = horizontalQ ? 0 : scale(q1), - y = verticalQ ? 0 : scale(extent[1]) - scale(q1), - t = 'translate(' + x + ',' + y + ')'; - return t; - }).attr('stroke', 'black').attr('stroke-width', whiskerStrokeWidth).attr('fill', 'none'); - - // set upper whisker (max) - uWhisk.transition().duration(transitionDuration).ease(easeFunc).attr('d', function (dd, ii) { - var dir = true, - x = 0, - y = 0, - h = horizontalQ ? scale(q4) - scale(q3) : objectSize, - w = verticalQ ? scale(q4) - scale(q3) : objectSize; - return whiskerPath(dir, x, y, w, h, whiskerWidthPercent, orient); - }).attr('transform', function (d, i) { - var x = horizontalQ ? 0 : scale(q3), - y = verticalQ ? 0 : scale(extent[1]) - scale(q4), - t = 'translate(' + x + ',' + y + ')'; - return t; - }).attr('stroke', 'black').attr('stroke-width', whiskerStrokeWidth).attr('fill', 'none'); - }); - - tooltip$$1.selection(container.selectAll('g:not(.to-remove).' + objectClass)).data(data); - tooltip$$1(); - } - - return boxwhisker; - } - - function selectFilter(selection) { - - var data, - namespace = 'd3sm-select-filter', - selectionName = 'Select options:', - defaultValue = undefined; - - var lastValue = undefined; - - selectFilter.data = function (_) { - return arguments.length ? (data = _, selectFilter) : data; - }; - selectFilter.namespace = function (_) { - return arguments.length ? (namespace = _, selectFilter) : namespace; - }; - selectFilter.selectionName = function (_) { - return arguments.length ? (selectionName = _, selectFilter) : selectionName; - }; - selectFilter.defaultValue = function (_) { - return arguments.length ? (defaultValue = _, selectFilter) : defaultValue; - }; - selectFilter.currentOption = currentOption; - - function selectFilter() { - var container = safeSelect(selection, 'div', 'input-group').classed(hypenate(namespace, 'container'), true), - selectPrepend = safeSelect(container, 'div', 'select-prepend').classed('input-group-prepend', true), - selectPrependSpan = safeSelect(selectPrepend, 'span', 'input-group-text').text(selectionName), - select = safeSelect(container, 'select', 'custom-select').classed(hypenate(namespace, 'select'), true), - selectAppend = safeSelect(container, 'div', 'select-append').classed('input-group-prepend', true), - selectAppendButton = safeSelect(selectAppend, 'a', 'filter-button').classed('btn btn-outline-secondary', true), - filterButtonIcon = safeSelect(selectAppendButton, 'i', 'fa fa-filter'), - inputGroup = safeSelect(container, 'div', 'filter-input-group').classed('input-group', true).classed('d-none', true), - inputPrepend = safeSelect(inputGroup, 'div', 'input-group-prepend'), - inputPrependSpan = safeSelect(inputPrepend, 'span', 'input-group-text').classed('search-button', true), - inputPrependSpanIcon = safeSelect(inputPrependSpan, 'i', 'fa fa-search'), - input = safeSelect(inputGroup, 'input', 'form-control').attr('placeholder', 'all').attr('type', 'text'), - inputAppend = safeSelect(inputGroup, 'div', 'input-group-append'), - inputAppendButton = safeSelect(inputAppend, 'a', 'close-button').classed('btn btn-outline-secondary', true), - inputAppendButtonIcon = safeSelect(inputAppendButton, 'i', 'fa fa-close'); - - var keys = d3.keys(data), - options = select.selectAll('option'); - - options = options.data(d3.keys(data)); - options = options.merge(options.enter().append('option')).attr('value', function (d, i) { - return d; - }).text(function (d, i) { - return d; - }); - - var filterButton = selectAppendButton, - closeButton = inputAppendButton; - - filterButton.on('click', function (d, i) { - var currentStyle = inputGroup.classed('d-none'); - inputGroup.classed('d-none', !currentStyle); - }); - - closeButton.on('click', function (d, i) { - input.property('value', '').dispatch('input'); - }); - - input.on('input', function (d, i) { - var val = input.property('value'), - reg = new RegExp(val, 'gi'), - use; - - if (val == '') { - use = keys; - } else { - use = []; - d3.keys(data).map(function (option, j) { - var match = option.match(reg); - if (match == null || match.join('') == '') ; else { - use.push(option); - } - }); - } - - options = select.selectAll('option'); - options = options.data(use); - options.exit().remove(); - options = options.merge(options.enter().append('option')).attr('value', function (d, i) { - return d; - }).text(function (d, i) { - return d; - }); - - var current = currentOption(); - if (lastValue != current) { - lastValue = current; - select.dispatch('change'); - } - }); - } - - function currentOption() { - var val = selection.select("select").property('value'); - return val == undefined || val == '' ? defaultValue == undefined ? d3.keys(data)[0] : defaultValue : val; - } - - return selectFilter; - } - - /******************************************************************************* - ** ** - ** ** - ** DATATOGGLE ** - ** ** - ** ** - *******************************************************************************/ - /** - * Creates a datatoggle - * @constructor datatoggle - * @param {d3.selection} selection - * @namespace datatoggle - * @returns {function} datatoggle - */ - function datatoggle(selection) { - var - /** - * What to do when a different key is clicked - * (see {@link datatoggle#updateFunction}) - * @param {function} [updateFunction=function(){}] - * @memberof datatoggle# - * @property - */ - updateFunction = function updateFunction() {}, - - /** - * Namespace for all items made by this instance of datatoggle - * @param {string} [namespace="d3sm-databar"] - * @memberof datatoggle# - * @property - */ - namespace = 'd3sm-databar', - - xAxisSelectQ = false, - xAxisOptions, - yAxisSelectQ = false, - yAxisOptions, - data; - toggle.xAxisSelectQ = function (_) { - return arguments.length ? (xAxisSelectQ = _, toggle) : xAxisSelectQ; - }; - toggle.yAxisSelectQ = function (_) { - return arguments.length ? (yAxisSelectQ = _, toggle) : yAxisSelectQ; - }; - toggle.xAxisOptions = function (_) { - return arguments.length ? (xAxisOptions = _, toggle) : xAxisOptions; - }; - toggle.yAxisOptions = function (_) { - return arguments.length ? (yAxisOptions = _, toggle) : yAxisOptions; - }; - toggle.data = function (_) { - return arguments.length ? (data = _, toggle) : data; - }; - - /** - * Gets / sets the updateFunction - * (see {@link datatoggle#updateFunction}) - * @function datatoggle.updateFunction - * @param {function} [_=none] - * @returns {datatoggle | function} - * @memberof datatoggle - * @property - * by default updateFunction = function(){} - */ - toggle.updateFunction = function (_) { - return arguments.length ? (updateFunction = _, toggle) : updateFunction; - }; - /** - * Gets / sets the namespace - * (see {@link datatoggle#namespace}) - * @function datatoggle.namespace - * @param {string} [_=none] - * @returns {datatoggle | string} - * @memberof datatoggle - * @property - * by default namespace = 'd3sm-databar' - */ - toggle.namespace = function (_) { - return arguments.length ? (namespace = _, toggle) : namespace; - }; - /** - * Gets / sets the currentKey - * (see {@link datatoggle#currentKey}) - * @function datatoggle.currentKey - * @param {string} [_=none] - * @returns {datatoggle | string} - * @memberof datatoggle - * @property - * by default currentKey = undefined - */ - toggle.currentKeys = function () { - var vals = {}; - d3.keys(filterSelects).map(function (k, i) { - vals[k] = filterSelects[k].currentOption(); - }); - return vals; - }; - - var filterSelects = {}; - function toggle() { - // selection options - - // selection.classed('d-flex flex-row', true) - // var filterButton = safeSelect(selection, 'a', 'slider-buttons') - // var filterI = safeSelect(filterButton, 'i', 'fa fa-sliders') - - /*BUG: unexpected behavior. - - when using bootstrap-eque way for collapse, clicking button submits to - the same page in applications (but not in demo), so using anchor () - - when using anchor, cause page to jump to that location - - when using show, the first open works, but the close does not. - */ - // filterButton.attr('class', 'btn btn-secondary') - // .attr('data-toggle', 'collapse') - // .attr('href', '#'+hypenate(namespace, 'data-toggle')) - // .attr('target', '_blank') - // .html(filterButton.html()+' Filters') - // .on('click', function(d, i){ - // d3.event.preventDefault() - // d3.event.stopPropagation() - // var dt = d3.select("#"+hypenate(namespace, 'data-toggle')) - // dt.classed('show', !dt.classed('show')) - // dt.classed('d-inline-flex', dt.classed('show')) - // - // filterButton.classed('btn-primary', dt.classed('show')) - // filterButton.classed('btn-secondary', !dt.classed('show')) - // }) - // .classed('d-inline-flex', true) - // .style("margin-right", '10px') - - - // var datatoggleCollapse = safeSelect(selection, 'div', hypenate(namespace,'collapse')) - // .attr('id', hypenate(namespace, 'data-toggle')) - // .classed('collapse', true) - - // var flexRow = safeSelect(datatoggleCollapse, 'div', 'd-inline-flex flex-row flex-wrap') - var flexRow = safeSelect(selection, 'div', 'd-inline-flex flex-row flex-wrap'); - - var dataopts = flexRow.selectAll('div.' + hypenate(namespace, 'select-filter')); - // remove excess - dataopts.exit().remove(); - // bind data - dataopts = dataopts.data(d3.keys(data)); - //enter - var doEnter = dataopts.enter().append('div').attr('class', 'select-filter'); - - dataopts = dataopts.merge(doEnter).style('margin-right', "10px"); - - dataopts.each(function (d, i) { - var t = d3.select(this); - var sf = selectFilter(t).data(data[d]).namespace(hypenate(namespace, d)).selectionName(d); - sf(); - filterSelects[d] = sf; - }); - - selection.selectAll('select').on('change', function () { - updateFunction(); - }); - // bind update function - // d3.selectAll - return toggle; - } - - return toggle; - } - - /******************************************************************************* - ** ** - ** ** - ** SCATTER ** - ** ** - ** ** - *******************************************************************************/ - /** - * Creates a scatter - * - * {@link https://sumneuron.gitlab.io/d3sm/demos/scatter/index.html Demo} - * @constructor scatter - * @param {d3.selection} selection - * @namespace scatter - * @returns {function} scatter - */ - function scatter(selection) { - - var - /** - * Data to plot. Assumed to be a object, where each key corresponds to a point - * (see {@link scatter#data}) - * @param {Object} [data=undefined] - * @memberof scatter# - * @property - */ - data, - - /** - * Amount of horizontal space (in pixels) avaible to render the scatter in - * (see {@link scatter#spaceX}) - * @param {number} [spaceX=undefined] - * @memberof scatter# - * @property - */ - spaceX, - - /** - * Amount of vertical space (in pixels) avaible to render the scatter in - * (see {@link scatter.spaceY}) - * @param {number} [spaceY=undefined] - * @memberof scatter# - * @property - */ - spaceY, - - - /** - * The scale for which scatter x values should be transformed by - * @param {d3.scale} [scaleX=d3.scaleLinear] - * @memberof scatter# - * @property - */ - scaleX = d3.scaleLinear(), - - /** - * The padding for the domain of the scaleX (see {@link scatter#scaleX}) - * @param {number} [domainPaddingX=0.5] - * @memberof scatter# - * @property - */ - domainPaddingX = 0.5, - - /** - * The function for getting the x value of the current point - * @param {function} [valueExtractorX=function(d, i){return data[d]['x']}] - * @memberof scatter# - * @property - */ - valueExtractorX = function valueExtractorX(d, i) { - return data[d]['x']; - }, - - - /** - * The scale for which scatter y values should be transformed by - * @param {d3.scale} [scaleY=d3.scaleLinear] - * @memberof scatter# - * @property - */ - scaleY = d3.scaleLinear(), - - /** - * The padding for the domain of the scaleY (see {@link scatter#scaleY}) - * @param {number} [domainPaddingY=0.5] - * @memberof scatter# - * @property - */ - domainPaddingY = 0.5, - - /** - * The function for getting the y value of the current point - * @param {function} [valueExtractorY=function(d, i){return data[d]['y']}] - * @memberof scatter# - * @property - */ - valueExtractorY = function valueExtractorY(d, i) { - return data[d]['y']; - }, - - - /** - * The scale for which scatter r values should be transformed by - * @param {d3.scale} [scaleR=d3.scaleLinear] - * @memberof scatter# - * @property - */ - scaleR = d3.scaleLinear(), - - /** - * The padding for the domain of the scaleR (see {@link scatter#scaleR}) - * @param {number} [domainPaddingR=0.5] - * @memberof scatter# - * @property - */ - domainPaddingR = 0.5, - - /** - * The function for getting the r value of the current point - * @param {function} [valueExtractorR=function(d, i){return 2}] - * @memberof scatter# - * @property - */ - valueExtractorR = function valueExtractorR(d, i) { - return 2; - }, - - /** - * The min radius a point can have - * @param {function} [minRadius=2] - * @memberof scatter# - * @property - */ - minRadius = 2, - - /** - * The min radius a point can have - * @param {function} [maxRadius=10] - * @memberof scatter# - * @property - */ - maxRadius = 10, - - - /** - * The stroke width of the points - * @param {number} [pointStrokeWidth=2] - * @memberof scatter# - * @property - */ - pointStrokeWidth = 2, - - /** - * Instance of ColorFunction - * @param {function} [colorFunction = colorFunction()] - * @memberof scatter# - * @property - */ - colorFunction$$1 = colorFunction(), - - /** - * Color of the background - * @param {string} [backgroundFill="transparent"] - * @memberof scatter# - * @property - */ - backgroundFill = 'transparent', - - /** - * Namespace for all items made by this instance of scatter - * @param {string} [namespace="d3sm-scatter"] - * @memberof scatter# - * @property - */ - namespace = 'd3sm-scatter', - - /** - * Class name for scatter container ( element) - * @param {string} [objectClass="scatter-point"] - * @memberof scatter# - * @property - */ - objectClass = 'scatter-point', - - /** - * Duration of all transitions of this element - * @param {number} [transitionDuration=1000] - * @memberof scatter# - * @property - */ - transitionDuration = 1000, - - /** - * Easing function for transitions - * @param {d3.ease} [easeFunc=d3.easeExp] - * @memberof scatter# - * @property - */ - easeFunc = d3.easeExp, - - - // useful values to extract to prevent re-calculation - /** - * The keys of the points - * @param {string[]} [pointKeys=undefined] - * @memberof scatter# - * @property - */ - pointKeys, - - /** - * The x values of the points - * @param {number[]} [valuesX=undefined] - * @memberof scatter# - * @property - */ - valuesX, - - /** - * The y values of the points - * @param {number[]} [valuesY=undefined] - * @memberof scatter# - * @property - */ - valuesY, - - /** - * The r values of the points - * @param {number[]} [valuesR=undefined] - * @memberof scatter# - * @property - */ - valuesR, - - - /** - * Instance of Tooltip - * @param {function} [tooltip=tooltip()] - * @memberof scatter# - * @property - */ - tooltip$$1 = tooltip(); - - /** - * Gets or sets the selection in which items are manipulated - * @param {d3.selection} [_=none] - * @returns {scatter | d3.selection} - * @memberof scatter - * @property - * by default selection = selection - */ - scatter.selection = function (_) { - return arguments.length ? (selection = _, scatter) : selection; - }; - /** - * Gets or sets the data - * (see {@link scatter#data}) - * @param {number} [_=none] - * @returns {scatter | object} - * @memberof scatter - * @property - */ - scatter.data = function (_) { - return arguments.length ? (data = _, scatter) : data; - }; - /** - * Gets or sets the amount of horizontal space in which items are manipulated - * (see {@link scatter#spaceX}) - * @param {number} [_=none] should be a number > 0 - * @returns {scatter | number} - * @memberof scatter - * @property - * by default spaceX = undefined - */ - scatter.spaceX = function (_) { - return arguments.length ? (spaceX = _, scatter) : spaceX; - }; - /** - * Gets or sets the amount of vertical space in which items are manipulated - * (see {@link scatter#spaceY}) - * @param {number} [_=none] should be a number > 0 - * @returns {scatter | number} - * @memberof scatter - * @property - * by default spaceY = undefined - */ - scatter.spaceY = function (_) { - return arguments.length ? (spaceY = _, scatter) : spaceY; - }; - - /** - * Gets / sets the x scale for which the scatter x values should be transformed by - * (see {@link scatter#scaleX}) - * @param {d3.scale} [_=none] - * @returns {scatter | d3.scale} - * @memberof scatter - * @property - * by default scaleX = d3.scaleLinear() - */ - scatter.scaleX = function (_) { - return arguments.length ? (scaleX = _, scatter) : scaleX; - }; - /** - * Gets / sets the padding for the domain of the x scale - * (see {@link scatter#domainPaddingX}) - * @param {number} [_=none] - * @returns {scatter | number} - * @memberof scatter - * @property - * by default domainPaddingX = 0.5 - */ - scatter.domainPaddingX = function (_) { - return arguments.length ? (domainPaddingX = _, scatter) : domainPaddingX; - }; - /** - * Gets / sets the valueExtractorX - * (see {@link scatter#valueExtractorX}) - * @param {function} [_=none] - * @returns {scatter | function} - * @memberof scatter - * @property - * by default valueExtractorX = function(key, index) { return data[key]['x'] } - */ - scatter.valueExtractorX = function (_) { - return arguments.length ? (valueExtractorX = _, scatter) : valueExtractorX; - }; - - /** - * Gets / sets the y scale for which the scatter y values should be transformed by - * (see {@link scatter#scaleY}) - * @param {d3.scale} [_=none] - * @returns {scatter | d3.scale} - * @memberof scatter - * @property - * by default scaleY = d3.scaleLinear() - */ - scatter.scaleY = function (_) { - return arguments.length ? (scaleY = _, scatter) : scaleY; - }; - /** - * Gets / sets the padding for the domain of the y scale - * (see {@link scatter#domainPaddingY}) - * @param {number} [_=none] - * @returns {scatter | number} - * @memberof scatter - * @property - * by default domainPaddingY = 0.5 - */ - scatter.domainPaddingY = function (_) { - return arguments.length ? (domainPaddingY = _, scatter) : domainPaddingY; - }; - /** - * Gets / sets the valueExtractorY - * (see {@link scatter#valueExtractorY}) - * @param {function} [_=none] - * @returns {scatter | function} - * @memberof scatter - * @property - * by default valueExtractorY = function(key, index) { return data[key]['y'] } - */ - scatter.valueExtractorY = function (_) { - return arguments.length ? (valueExtractorY = _, scatter) : valueExtractorY; - }; - - /** - * Gets / sets the r scale for which the scatter r values should be transformed by - * (see {@link scatter#scaleR}) - * @param {d3.scale} [_=none] - * @returns {scatter | d3.scale} - * @memberof scatter - * @property - * by default scaleR = d3.scaleLinear() - */ - scatter.scaleR = function (_) { - return arguments.length ? (scaleR = _, scatter) : scaleR; - }; - /** - * Gets / sets the padding for the domain of the r scale - * (see {@link scatter#domainPaddingR}) - * @param {number} [_=none] - * @returns {scatter | number} - * @memberof scatter - * @property - * by default domainPaddingR = 0.5 - */ - scatter.domainPaddingR = function (_) { - return arguments.length ? (domainPaddingR = _, scatter) : domainPaddingR; - }; - /** - * Gets / sets the valueExtractorR - * (see {@link scatter#valueExtractorR}) - * @param {function} [_=none] - * @returns {scatter | function} - * @memberof scatter - * @property - * by default valueExtractorR = function(key, index) { return data[key]['r'] } - */ - scatter.valueExtractorR = function (_) { - return arguments.length ? (valueExtractorR = _, scatter) : valueExtractorR; - }; - /** - * Gets / sets the minRadius - * (see {@link scatter#minRadius}) - * @param {number} [_=none] - * @returns {scatter | number} - * @memberof scatter - * @property - * by default minRadius = 2 - */ - scatter.minRadius = function (_) { - return arguments.length ? (minRadius = _, scatter) : minRadius; - }; - /** - * Gets / sets the maxRadius - * (see {@link scatter#maxRadius}) - * @param {number} [_=none] - * @returns {scatter | number} - * @memberof scatter - * @property - * by default maxRadius = 10 - */ - scatter.maxRadius = function (_) { - return arguments.length ? (maxRadius = _, scatter) : maxRadius; - }; - - /** - * Gets / sets the pointStrokeWidth - * (see {@link scatter#pointStrokeWidth}) - * @param {number} [_=none] - * @returns {scatter | number} - * @memberof scatter - * @property - * by default pointStrokeWidth = 2 - */ - scatter.pointStrokeWidth = function (_) { - return arguments.length ? (pointStrokeWidth = _, scatter) : pointStrokeWidth; - }; - /** - * Gets / sets the colorFunction - * (see {@link scatter#colorFunction}) - * @param {number} [_=none] - * @returns {scatter | number} - * @memberof scatter - * @property - * by default colorFunction = colorFunction() - */ - scatter.colorFunction = function (_) { - return arguments.length ? (colorFunction$$1 = _, scatter) : colorFunction$$1; - }; - /** - * Gets / sets the backgroundFill - * (see {@link scatter#backgroundFill}) - * @param {string} [_=none] - * @returns {scatter | string} - * @memberof scatter - * @property - * by default backgroundFill = 'transparent' - */ - scatter.backgroundFill = function (_) { - return arguments.length ? (backgroundFill = _, scatter) : backgroundFill; - }; - /** - * Gets / sets the namespace - * (see {@link scatter#namespace}) - * @param {string} [_=none] - * @returns {scatter | string} - * @memberof scatter - * @property - * by default namespace = 'd3sm-scatter' - */ - scatter.namespace = function (_) { - return arguments.length ? (namespace = _, scatter) : namespace; - }; - /** - * Gets / sets the objectClass - * (see {@link scatter#objectClass}) - * @param {string} [_=none] - * @returns {scatter | string} - * @memberof scatter - * @property - * by default objectClass = 'tick-group' - */ - scatter.objectClass = function (_) { - return arguments.length ? (objectClass = _, scatter) : objectClass; - }; - /** - * Gets / sets the transitionDuration - * (see {@link scatter#transitionDuration}) - * @param {number} [_=none] - * @returns {scatter | number} - * @memberof scatter - * @property - * by default transitionDuration = 1000 - */ - scatter.transitionDuration = function (_) { - return arguments.length ? (transitionDuration = _, scatter) : transitionDuration; - }; - /** - * Gets / sets the easeFunc - * (see {@link scatter#easeFunc}) - * @param {d3.ease} [_=none] - * @returns {scatter | d3.ease} - * @memberof scatter - * @property - * by default easeFunc = d3.easeExp - */ - scatter.easeFunc = function (_) { - return arguments.length ? (easeFunc = _, scatter) : easeFunc; - }; - - /** - * Gets / sets the pointKeys - * (see {@link scatter#pointKeys}) - * @param {string[]} [_=none] - * @returns {scatter | string[]} - * @memberof scatter - * @property - * by default pointKeys = undefined - */ - scatter.pointKeys = function (_) { - return arguments.length ? (pointKeys = _, scatter) : pointKeys; - }; - /** - * Gets / sets the valuesX - * (see {@link scatter#valuesX}) - * @param {number[]} [_=none] - * @returns {scatter | number[]} - * @memberof scatter - * @property - * by default valuesX = undefined - */ - scatter.valuesX = function (_) { - return arguments.length ? (valuesX = _, scatter) : valuesX; - }; - /** - * Gets / sets the valuesY - * (see {@link scatter#valuesY}) - * @param {number[]} [_=none] - * @returns {scatter | number[]} - * @memberof scatter - * @property - * by default valuesY = undefined - */ - scatter.valuesY = function (_) { - return arguments.length ? (valuesY = _, scatter) : valuesY; - }; - /** - * Gets / sets the valuesR - * (see {@link scatter#valuesR}) - * @param {number[]} [_=none] - * @returns {scatter | number[]} - * @memberof scatter - * @property - * by default valuesR = undefined - */ - scatter.valuesR = function (_) { - return arguments.length ? (valuesR = _, scatter) : valuesR; - }; - /** - * Gets / sets the tooltip - * (see {@link scatter#tooltip}) - * @param {tooltip} [_=none] - * @returns {scatter | tooltip} - * @memberof scatter - * @property - * by default tooltip = tooltip() - */ - - scatter.tooltip = function (_) { - return arguments.length ? (tooltip$$1 = _, scatter) : tooltip$$1; - }; - - function scatter() { - // background cliping rectangle - var bgcpRect = { x: 0, y: 0, width: spaceX, height: spaceY }; - var container = setupContainer(selection, namespace, bgcpRect, backgroundFill); - - pointKeys = d3.keys(data); - valuesX = pointKeys.map(valueExtractorX); - valuesY = pointKeys.map(valueExtractorY); - valuesR = pointKeys.map(valueExtractorR); - - var numberOfObjects = pointKeys.length; - var extentX = [Math.min.apply(Math, toConsumableArray(valuesX)) - domainPaddingX, Math.max.apply(Math, toConsumableArray(valuesX)) + domainPaddingX]; - var extentY = [Math.min.apply(Math, toConsumableArray(valuesY)) - domainPaddingY, Math.max.apply(Math, toConsumableArray(valuesY)) + domainPaddingY]; - var extentR = [Math.min.apply(Math, toConsumableArray(valuesR)) - domainPaddingR, Math.max.apply(Math, toConsumableArray(valuesR)) + domainPaddingR]; - - scaleX.domain(extentX).range([0, spaceX]); - scaleY.domain(extentY).range([spaceY, 0]); - scaleR.domain(extentR).range([minRadius, maxRadius]); - - var points = container.selectAll('.' + objectClass); - points = points.data(pointKeys); - var pEnter = points.enter().append('circle').attr('class', objectClass).attr('cx', 0).attr('cy', spaceY).attr('r', 0); - - var pExit = points.exit(); - - points = points.merge(pEnter); - - points.each(function (key, i) { - var t = d3.select(this), - currentData = data[key], - x = valuesX[i], - y = valuesY[i], - r = valuesR[i], - fillColor = colorFunction$$1(key, currentData, i, 'fill'), - strokeColor = colorFunction$$1(key, currentData, i, 'stroke'); - - t.transition().duration(transitionDuration).ease(easeFunc).attr('cx', scaleX(x)).attr('cy', scaleY(y)).attr('r', scaleR(r)).attr('fill', fillColor).attr('stroke', strokeColor).attr('stroke-width', pointStrokeWidth); - - t.on('mouseover', function (d, i) { - points.style('opacity', 0.2); - t.style('opacity', 1); - t.transition().duration(transitionDuration / 2).ease(easeFunc).attr('stroke-width', pointStrokeWidth * 2).attr('r', scaleR(r) * 1.5); - }); - t.node().addEventListener('mouseout', function () { - container.selectAll('.' + objectClass).style('opacity', 1); - t.transition().duration(transitionDuration / 2).ease(easeFunc).attr('stroke-width', pointStrokeWidth).attr('r', scaleR(r)); - }); - }); - - pExit.transition().duration(transitionDuration).ease(easeFunc).attr('cx', 0).attr('cy', spaceY).attr('r', 0).remove(); - - tooltip$$1.selection(points).data(data); - - tooltip$$1(); - } - - return scatter; - } - - /******************************************************************************* - - ** ** - ** ** - ** PLOTZOOM ** - ** ** - ** ** - *******************************************************************************/ - /** - * Creates an plotZoom instance, which can handle drag and scroll events - * @constructor plotZoom - * @param {function} chart a function instance of one of the d3sm plots (e.g. bar, boxwhisker, bubbleHeatmap, violin, etc) - * @param {axis} xAxis the axis instance responsible for the x axis - * @param {axis} yAxis the axis instance responsible for the y axis - * @namespace plotZoom - * @returns {function} zoom - */ - function plotZoom(chart, xAxis, yAxis) { - var - /** - * The event on which to fire - * (see {@link plotZoom#eventType}) - * @param {string} eventType which event it should handle. Currently supports scroll and wheel - * @memberof plotZoom# - * @property - */ - eventType, - - /** - * A scaling factor for the wheel "speed" - * (see {@link plotZoom#wheelSpeed}) - * @param {number} [wheelSpeed=20] scales the wheel translation by wheelSpeed - * @memberof plotZoom# - * @property - */ - wheelSpeed = 20, - - /** - * The orientation in which to allow scrolling: 'horizontal', 'vertical', or '2D' - * (see {@link plotZoom#orient}) - * @param {string} [orient=chart.orient() || 'horizontal'] - * @memberof plotZoom# - * @property - */ - orient = chart.orient == undefined ? 'horizontal' : chart.orient(), - - /** - * The max distance allowed to scroll in the x direction - * (see {@link plotZoom#xLock}) - * @param {number} [xLock=chart.spaceX()] ideally chart.overflowQ() == true and this value is the - * bounding rect across all elements in the chart minus the space in which to show. - * @memberof plotZoom# - * @property - */ - xLock = chart.spaceX(), - - /** - * The max distance allowed to scroll in the y direction - * (see {@link plotZoom#yLock}) - * @param {number} [yLock=chart.spaceY()] ideally chart.overflowQ() == true and this value is the - * bounding rect across all elements in the chart minus the space in which to show. - * @memberof plotZoom# - * @property - */ - yLock = chart.spaceY(), - chartSel = chart.selection(), - xAxisSel = xAxis.selection(), - yAxisSel = yAxis.selection(), - svg = d3.select(chartSel.thisSVG()); - - /** - * Gets or sets the event type in which to respond - * (see {@link plotZoom#eventType}) - * @param {string} [_=none] should be 'drag' or 'wheel' - * @returns {zoom | string} - * @memberof plotZoom - * @property - * by default plotZoom=undefined - */ - zoom.eventType = function (_) { - return arguments.length ? (eventType = _, zoom) : eventType; - }; - /** - * Gets or sets the wheel speed in which to scale the wheel scroll transform - * (see {@link plotZoom#wheelSpeed}) - * @param {number} [_=none] - * @returns {zoom | number} - * @memberof plotZoom - * @property - * by default wheelSpeed=20 - */ - zoom.wheelSpeed = function (_) { - return arguments.length ? (wheelSpeed = _, zoom) : wheelSpeed; - }; - /** - * Gets or sets the orientation in which items are manipulated - * (see {@link plotZoom#orient}) - * @param {string} [_=none] should be horizontal, vertical, or 2D - * @returns {zoom | string} - * @memberof plotZoom - * @property - * by default orient=chart.orient() || 'horizontal' - */ - zoom.orient = function (_) { - return arguments.length ? (orient = _, zoom) : orient; - }; - - /** - * Gets or sets the max distance in which to scroll X - * (see {@link plotZoom#xLock}) - * @param {number} [_=none] should be a positive value - * @returns {zoom | number} - * @memberof plotZoom - * @property - * by default xLock=chart.spaceX() - */ - zoom.xLock = function (_) { - return arguments.length ? (xLock = _, zoom) : xLock; - }; - /** - * Gets or sets the max distance in which to scroll Y - * (see {@link plotZoom#yLock}) - * @param {number} [_=none] should be a positive value - * @returns {zoom | number} - * @memberof plotZoom - * @property - * by default yLock=chart.spaceY() - */ - zoom.yLock = function (_) { - return arguments.length ? (yLock = _, zoom) : yLock; - }; - - function setLocks() { - var chartObjSel = chartSel.select('.' + hypenate(chart.namespace(), 'object-container')); - var chartObjTrans = getTranslation(chartObjSel.attr('transform')); - var cos = chartObjSel.attr('transform', 'translate(0,0)'); - xLock = chartSel.node().getBBox().width - chart.spaceX() * .9; - yLock = chartSel.node().getBBox().height - chart.spaceY() * .9; - cos.attr('transform', 'translate(' + chartObjTrans[0] + ',' + chartObjTrans[1] + ')'); - log('plotZoom', 'setLocks', { xLock: xLock, yLock: yLock }); - } - - /** - * Sets the x and y locks (how far one can scroll) - * (see {@link plotZoom#xLock} and {@link plotZoom#yLock}) - * @function plotZoom.setLocks - * @returns {undefined} - * @memberof plotZoom - * @property - */ - zoom.setLocks = setLocks; - - function zoom() { - setLocks(); - - var horizontalQ, verticalQ; - if (orient == '2D') { - horizontalQ = true;verticalQ = true; - } - if (orient == 'horizontal') { - horizontalQ = true;verticalQ = false; - } - if (orient == 'vertical') { - verticalQ = true;horizontalQ = false; - } - - // capture transform event - var transform = d3.event.transform; - - var chartBox = chartSel.node().getBBox(); - var xAxisBox = xAxisSel.node().getBBox(); - var yAxisBox = xAxisSel.node().getBBox(); - - var chartWidth = chartBox.width - chartBox.x; - var chartHeight = chartBox.height - chartBox.y; - var xAxisWidth = xAxisBox.width; // - xAxisBox.x - var xAxisHeight = xAxisBox.height; // - xAxisBox.y - var yAxisWidth = yAxisBox.width; // - yAxisBox.x - var yAxisHeight = yAxisBox.height; // -yAxisBox.y - - // enable wheel event - if (eventType == "wheel") { - var e = d3.event; - // prevent page scrolling - e.preventDefault(); - // event delta is very very slow, so speed it up with wheel speed - var w = d3.event.deltaY * wheelSpeed; - var shiftQ = d3.event.shiftKey; - - // d3 has no way to make custom transform, so make an object and add - // the necessary functions to make wheel event compatible with drag events - if (orient == '2D') { - transform = shiftQ ? { k: 1, x: w, y: 0 } : { k: 1, x: 0, y: w }; - } else { - transform = horizontalQ ? { k: 1, x: w, y: 0 } : { k: 1, x: 0, y: w }; - } - // the * -1 inverts the direction - transform.applyX = function (x) { - return x * this.k + this.x * -1; - }; - transform.applyY = function (y) { - return y * this.k + this.y * -1; - }; - } - - var chartObjSel = chartSel.select('.' + hypenate(chart.namespace(), 'object-container')); - var xAxisObjSel = xAxisSel.select('.' + hypenate(xAxis.namespace(), 'object-container')); - var yAxisObjSel = yAxisSel.select('.' + hypenate(yAxis.namespace(), 'object-container')); - - // xLock = chartSel.node().getBBox().width - chart.spaceX() - chartSel.node().getBBox().x - // yLock = chartSel.node().getBBox().height - chart.spaceY() - // console.table({'xLock':xLock, "yLock":yLock}) - // bhm.selection().node().getBBox().width - bhm.spaceX() - - - var chartObjTrans = getTranslation(chartObjSel.attr('transform')); - var xAxisObjTrans = getTranslation(xAxisObjSel.attr('transform')); - var yAxisObjTrans = getTranslation(yAxisObjSel.attr('transform')); - - var x = horizontalQ ? transform.applyX(chartObjTrans[0]) : 0; - if (horizontalQ) { - x = x < -xLock ? (transform.x = 0, -xLock) : (transform.x = 0, Math.min(x, 0)); - } - - var y = verticalQ ? transform.applyY(chartObjTrans[1]) : 0; - if (verticalQ) { - y = y < -yLock ? (transform.y = 0, -yLock) : (transform.y = 0, Math.min(y, 0)); - } - - chartObjSel.attr('transform', 'translate(' + x + ',' + y + ')'); - if (horizontalQ) { - xAxisObjSel.attr('transform', 'translate(' + x + ',' + 0 + ')'); - } - if (verticalQ) { - yAxisObjSel.attr('transform', 'translate(' + 0 + ',' + y + ')'); - } - - // var lasso = svg.select(".lasso-container") - // if (!lasso.empty()) { - // lasso.attr('transform', 'translate('+x+','+y+')') - // } - } - - zoom.reset = function () { - - var chartObjSel = chartSel.select('.' + hypenate(chart.namespace(), 'object-container')); - var xAxisObjSel = xAxisSel.select('.' + hypenate(xAxis.namespace(), 'object-container')); - var yAxisObjSel = yAxisSel.select('.' + hypenate(yAxis.namespace(), 'object-container')); - chartObjSel.attr('transform', 'translate(' + 0 + ',' + 0 + ')'); - xAxisObjSel.attr('transform', 'translate(' + 0 + ',' + 0 + ')'); - yAxisObjSel.attr('transform', 'translate(' + 0 + ',' + 0 + ')'); - }; - - return zoom; - } - - /******************************************************************************* - - ** ** - ** ** - ** PLOTZOOM ** - ** ** - ** ** - *******************************************************************************/ - /** - * Creates an plotZoom instance, which can handle drag and scroll events - * @constructor plotZoom - * @param {function} chart a function instance of one of the d3sm plots (e.g. bar, boxwhisker, bubbleHeatmap, violin, etc) - * @param {axis} xAxis the axis instance responsible for the x axis - * @param {axis} yAxis the axis instance responsible for the y axis - * @namespace plotZoom - * @returns {function} zoom - */ - function multiPlotZoom(chart) { - var - /** - * The event on which to fire - * (see {@link plotZoom#eventType}) - * @param {string} eventType which event it should handle. Currently supports scroll and wheel - * @memberof plotZoom# - * @property - */ - eventType, - - /** - * A scaling factor for the wheel "speed" - * (see {@link plotZoom#wheelSpeed}) - * @param {number} [wheelSpeed=20] scales the wheel translation by wheelSpeed - * @memberof plotZoom# - * @property - */ - wheelSpeed = 20, - - /** - * The orientation in which to allow scrolling: 'horizontal', 'vertical', or '2D' - * (see {@link plotZoom#orient}) - * @param {string} [orient=chart.orient() || 'horizontal'] - * @memberof plotZoom# - * @property - */ - orient = chart.orient == undefined ? 'horizontal' : chart.orient(), - - /** - * The max distance allowed to scroll in the x direction - * (see {@link plotZoom#xLock}) - * @param {number} [xLock=chart.spaceX()] ideally chart.overflowQ() == true and this value is the - * bounding rect across all elements in the chart minus the space in which to show. - * @memberof plotZoom# - * @property - */ - xLock = chart.spaceX(), - - /** - * The max distance allowed to scroll in the y direction - * (see {@link plotZoom#yLock}) - * @param {number} [yLock=chart.spaceY()] ideally chart.overflowQ() == true and this value is the - * bounding rect across all elements in the chart minus the space in which to show. - * @memberof plotZoom# - * @property - */ - yLock = chart.spaceY(), - chartSel = chart.selection(), - svg = d3.select(chartSel.thisSVG()), - xComponents = [], - yComponents = []; - - /** - * Gets or sets the event type in which to respond - * (see {@link plotZoom#eventType}) - * @param {string} [_=none] should be 'drag' or 'wheel' - * @returns {zoom | string} - * @memberof plotZoom - * @property - * by default plotZoom=undefined - */ - zoom.eventType = function (_) { - return arguments.length ? (eventType = _, zoom) : eventType; - }; - /** - * Gets or sets the wheel speed in which to scale the wheel scroll transform - * (see {@link plotZoom#wheelSpeed}) - * @param {number} [_=none] - * @returns {zoom | number} - * @memberof plotZoom - * @property - * by default wheelSpeed=20 - */ - zoom.wheelSpeed = function (_) { - return arguments.length ? (wheelSpeed = _, zoom) : wheelSpeed; - }; - /** - * Gets or sets the orientation in which items are manipulated - * (see {@link plotZoom#orient}) - * @param {string} [_=none] should be horizontal, vertical, or 2D - * @returns {zoom | string} - * @memberof plotZoom - * @property - * by default orient=chart.orient() || 'horizontal' - */ - zoom.orient = function (_) { - return arguments.length ? (orient = _, zoom) : orient; - }; - - /** - * Gets or sets the max distance in which to scroll X - * (see {@link plotZoom#xLock}) - * @param {number} [_=none] should be a positive value - * @returns {zoom | number} - * @memberof plotZoom - * @property - * by default xLock=chart.spaceX() - */ - zoom.xLock = function (_) { - return arguments.length ? (xLock = _, zoom) : xLock; - }; - /** - * Gets or sets the max distance in which to scroll Y - * (see {@link plotZoom#yLock}) - * @param {number} [_=none] should be a positive value - * @returns {zoom | number} - * @memberof plotZoom - * @property - * by default yLock=chart.spaceY() - */ - zoom.yLock = function (_) { - return arguments.length ? (yLock = _, zoom) : yLock; - }; - - zoom.xComponents = function (_) { - return arguments.length ? (xComponents = _, zoom) : xComponents; - }; - zoom.yComponents = function (_) { - return arguments.length ? (yComponents = _, zoom) : yComponents; - }; - - function setLocks() { - var chartObjSel = chartSel.select('.' + hypenate(chart.namespace(), 'object-container')); - var chartObjTrans = getTranslation(chartObjSel.attr('transform')); - var cos = chartObjSel.attr('transform', 'translate(0,0)'); - - xLock = chartSel.node().getBBox().width - chart.spaceX(); // * .9 - yLock = chartSel.node().getBBox().height - chart.spaceY(); // * .9 - cos.attr('transform', 'translate(' + chartObjTrans[0] + ',' + chartObjTrans[1] + ')'); - log('plotZoom', 'setLocks', { xLock: xLock, yLock: yLock }); - } - - /** - * Sets the x and y locks (how far one can scroll) - * (see {@link plotZoom#xLock} and {@link plotZoom#yLock}) - * @function plotZoom.setLocks - * @returns {undefined} - * @memberof plotZoom - * @property - */ - zoom.setLocks = setLocks; - - function zoom() { - setLocks(); - - var xComponentsSel = xComponents.map(function (d, i) { - return d.selection(); - }), - yComponentsSel = yComponents.map(function (d, i) { - return d.selection(); - }); - - var horizontalQ, verticalQ; - if (orient == '2D') { - horizontalQ = true;verticalQ = true; - } - if (orient == 'horizontal') { - horizontalQ = true;verticalQ = false; - } - if (orient == 'vertical') { - verticalQ = true;horizontalQ = false; - } - - // capture transform event - var transform = d3.event.transform; - - var chartBox = chartSel.node().getBBox(); - var xComponentsBox = xComponentsSel.map(function (d, i) { - return d.node().getBBox(); - }); - var yComponentsBox = xComponentsSel.map(function (d, i) { - return d.node().getBBox(); - }); - - var chartWidth = chartBox.width - chartBox.x; - var chartHeight = chartBox.height - chartBox.y; - - // enable wheel event - if (eventType == "wheel") { - var e = d3.event; - // prevent page scrolling - e.preventDefault(); - // event delta is very very slow, so speed it up with wheel speed - var w = d3.event.deltaY * wheelSpeed; - var shiftQ = d3.event.shiftKey; - - // d3 has no way to make custom transform, so make an object and add - // the necessary functions to make wheel event compatible with drag events - if (orient == '2D') { - transform = shiftQ ? { k: 1, x: w, y: 0 } : { k: 1, x: 0, y: w }; - } else { - transform = horizontalQ ? { k: 1, x: w, y: 0 } : { k: 1, x: 0, y: w }; - } - // * -1 is invert - transform.applyX = function (x) { - return x * this.k + this.x * -1; - }; - transform.applyY = function (y) { - return y * this.k + this.y * -1; - }; - } - - var chartObjSel = chartSel.select('.' + hypenate(chart.namespace(), 'object-container')); - var xComponentObjSel = xComponentsSel.map(function (d, i) { - return d.select('.' + hypenate(xComponents[i].namespace(), 'object-container')); - }); - var yComponentObjSel = yComponentsSel.map(function (d, i) { - return d.select('.' + hypenate(yComponents[i].namespace(), 'object-container')); - }); - - var chartObjTrans = getTranslation(chartObjSel.attr('transform')); - var xComponentsObjTrans = xComponentObjSel.map(function (d, i) { - return getTranslation(d.attr('transform')); - }); - var yComponentsObjTrans = yComponentObjSel.map(function (d, i) { - return getTranslation(d.attr('transform')); - }); - - var x = horizontalQ ? transform.applyX(chartObjTrans[0]) : 0; - if (horizontalQ) { - x = x < -xLock ? (transform.x = 0, -xLock) : (transform.x = 0, Math.min(x, 0)); - } - - var y = verticalQ ? transform.applyY(chartObjTrans[1]) : 0; - if (verticalQ) { - y = y < -yLock ? (transform.y = 0, -yLock) : (transform.y = 0, Math.min(y, 0)); - } - - chartObjSel.attr('transform', 'translate(' + x + ',' + y + ')'); - if (horizontalQ) { - // xAxisObjSel.attr('transform', 'translate('+x+','+0+')') - xComponentObjSel.map(function (d, i) { - d.attr('transform', 'translate(' + x + ',' + 0 + ')'); - }); - } - if (verticalQ) { - // yAxisObjSel.attr('transform', 'translate('+0+','+y+')') - yComponentObjSel.map(function (d, i) { - d.attr('transform', 'translate(' + 0 + ',' + y + ')'); - }); - } - } - - zoom.reset = function () { - - var chartObjSel = chartSel.select('.' + hypenate(chart.namespace(), 'object-container')); - var xAxisObjSel = xAxisSel.select('.' + hypenate(xAxis.namespace(), 'object-container')); - var yAxisObjSel = yAxisSel.select('.' + hypenate(yAxis.namespace(), 'object-container')); - chartObjSel.attr('transform', 'translate(' + 0 + ',' + 0 + ')'); - xAxisObjSel.attr('transform', 'translate(' + 0 + ',' + 0 + ')'); - yAxisObjSel.attr('transform', 'translate(' + 0 + ',' + 0 + ')'); - }; - - return zoom; - } - - /******************************************************************************* - ** ** - ** ** - ** VIOLIN ** - ** ** - ** ** - *******************************************************************************/ - - /** - * Creates a violin - * - * {@link https://sumneuron.gitlab.io/d3sm/demos/basic-violins/index.html Demo} - * @constructor violin - * @param {d3.selection} selection - * @namespace violin - * @returns {function} violin - */ - function violin(selection) { - var - /** - * Data to plot. Assumed to be a object, where each key corresponds to a violin - * (see {@link violin#data}) - * @param {Object} [data=undefined] - * @memberof violin# - * @property - */ - data, - - /** - * Which direction to render the bars in - * (see {@link violin#orient}) - * @param {number} [orient='horizontal'] - * @memberof violin# - * @property - */ - orient = 'horizontal', - - /** - * Amount of horizontal space (in pixels) avaible to render the violin in - * (see {@link violin#spaceX}) - * @param {number} [spaceX=undefined] - * @memberof violin# - * @property - */ - spaceX, - - /** - * Amount of vertical space (in pixels) avaible to render the violin in - * (see {@link violin.spaceY}) - * @param {number} [spaceY=undefined] - * @memberof violin# - * @property - */ - spaceY, - - /** - * Whether or not to allow violin to render elements pass the main spatial dimension - * given the orientation (see {@link violin#orient}), where {@link violin#orient}="horizontal" - * the main dimension is {@link violin#spaceX} and where {@link violin#orient}="vertical" - * the main dimension is {@link violin#spaceY} - * @param {boolean} [overflowQ=false] - * @memberof violin# - * @property - */ - overflowQ = true, - - /** - * Whether or not to display points inside the points - * @param {boolean} [pointsQ=false] - * @memberof violin# - * @property - */ - pointsQ = true, - - /** - * An array - putatively of other arrays - depicting how bars should be arranged - * @param {Array[]} [grouping=undefined] - * @memberof violin# - * @property - */ - grouping, - - /** - * How to get the value of the violin - * @param {function} [valueExtractor=function(key, index) { return data[key] }] - * @memberof violin# - * @property - */ - valueExtractor = function valueExtractor(key, index) { - return data[key]; - }, - - /** - * How to sort the bars - if {@link violin#grouping} is not provided. - * @param {function} [sortingFunction=function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}] - * @memberof violin# - * @property - */ - sortingFunction = function sortingFunction(keyA, keyB) { - return d3.descending(data[keyA], data[keyB]); - }, - - - /** - * The scale for which violin values should be transformed by - * @param {d3.scale} [scale=d3.scaleLinear] - * @memberof violin# - * @property - */ - scale = d3.scaleLinear(), - - /** - * The padding for the domain of the scale (see {@link violin#scale}) - * @param {number} [domainPadding=0.5] - * @memberof violin# - * @property - */ - domainPadding = 0.5, - - /** - * Default space for the spacer (percentage) of main dimension given the orientation - * (see {@link violin#orient}), where {@link violin#orient}="horizontal" - * the main dimension is {@link violin#spaceX} and where {@link violin#orient}="vertical" - * the main dimension is {@link violin#spaceY} between bars - * @param {number} [objectSpacer=0.05] - * @memberof violin# - * @property - */ - objectSpacer = 0.05, - - /** - * The minimum size that an object can be - * @param {number} [minObjectSize=50] - * @memberof violin# - * @property - */ - minObjectSize = 50, - - /** - * The maximum size that an object can be - * @param {number} [maxObjectSize=100] - * @memberof violin# - * @property - */ - maxObjectSize = 100, - - - /** - * The stroke width of the bars - * @param {number} [barStrokeWidth=2] - * @memberof violin# - * @property - */ - objectStrokeWidth = 2, - - /** - * Instance of ColorFunction - * @param {function} [colorFunction = colorFunction()] - * @memberof violin# - * @property - */ - colorFunction$$1 = colorFunction(), - - /** - * Instance of ColorFunction modified by a scale for the points - * @param {function} [pointColorFunc = colorFunction()] - * @memberof violin# - * @property - */ - pointColorFunc = function pointColorFunc(d, type, base, min, max) { - var minMaxHexScale = d3.scaleLinear().domain([min, max]).range([-0.25, 0.05]); - var scaledColor = modifyHexidecimalColorLuminance(base.replace('#', ''), minMaxHexScale(d)); - var mod = type == "stroke" ? 0 : 0.25; - return modifyHexidecimalColorLuminance(scaledColor.replace('#', ''), mod); - }, - - - /** - * The radius of a point - * @param {number} [pointRadius=3] - * @memberof violin# - * @property - */ - pointRadius = 3, - - /** - * The stroke width of the oints - * @param {number} [pointStrokeWidth=2] - * @memberof violin# - * @property - */ - pointStrokeWidth = 2, - - - /** - * Color of the background - * @param {string} [backgroundFill="transparent"] - * @memberof violin# - * @property - */ - backgroundFill = 'transparent', - - /** - * Namespace for all items made by this instance of violin - * @param {string} [namespace="d3sm-violin"] - * @memberof violin# - * @property - */ - namespace = 'd3sm-violin', - - /** - * Class name for violin container ( element) - * @param {string} [objectClass="violin"] - * @memberof violin# - * @property - */ - objectClass = 'violin', - - /** - * Duration of all transitions of this element - * @param {number} [transitionDuration=1000] - * @memberof violin# - * @property - */ - transitionDuration = 1000, - - /** - * Easing function for transitions - * @param {d3.ease} [easeFunc=d3.easeExp] - * @memberof violin# - * @property - */ - easeFunc = d3.easeExp, - - - /** - * The keys corresponding to each quartile - * @param {string[]} [quartileKeys=["Q0", "Q1", "Q2", "Q3", "Q4"]] - * @memberof violin# - * @property - */ - quartileKeys = ["Q0", "Q1", "Q2", "Q3", "Q4"], - - - /** - * The keys of the bars - * @param {string[]} [violinKeys=undefined] - * @memberof violin# - * @property - */ - violinKeys, - - /** - * The values of the bars - * @param {number[]} [violinValues=undefined] - * @memberof violin# - * @property - */ - violinValues, - - /** - * The objectSize (actual width) used by the bars - * @param {number} [objectSize=undefined] - * @memberof violin# - * @property - */ - objectSize, - - /** - * The spacerSize (actual width) used by the spacers between the bars - * @param {number} [spacerSize=undefined] - * @memberof violin# - * @property - */ - spacerSize, - - - /** - * Instance of Tooltip - * @param {function} [tooltip=tooltip()] - * @memberof violin# - * @property - */ - tooltip$$1 = tooltip().keys([quartileKeys[4], quartileKeys[3], quartileKeys[2], quartileKeys[1], quartileKeys[0]]), - pointsTooltip = tooltip(), - - // pointKeyExtractor = function(violinKey, violinData, violinValues) {return d3.keys(violinValues[violinKey].values)}, - // pointValueExtractor = function(pointKey, violinKey, violinData, violinValues) {return violinValues[violinKey].values[pointKey]}, - - - /** - * Function which given the key of the violin and that key's associated value - * returns the object consiting of the points of the violin - * (see {@link violin#violinPointsExtractor}) - * @param {Object} [violinPointsExtractor=function(violinKey, violinData) { return violinData.points }] - * @memberof violin# - * @property - */ - violinPointsExtractor = function violinPointsExtractor(violinKey, violinData) { - return violinData.points; - }, - - /** - * Function which given the key of the current point and the object of points for the - * violin, returns the numerical value of the point - * (see {@link violin#violinPointValueExtractor}) - * @param {Object} [violinPointValueExtractor=function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value }] - * @memberof violin# - * @property - */ - violinPointValueExtractor = function violinPointValueExtractor(violinPointKey, violinPointData) { - return violinPointData[violinPointKey].value; - }; - - /** - * Gets or sets the violinPointsExtractor - * @param {function} [_=none] - * @returns {violin | function} - * @memberof violin - * @property - * by default violinPointsExtractor = function(violinKey, violinData) { return violinData.points } - */ - violin.violinPointsExtractor = function (_) { - return arguments.length ? (violinPointsExtractor = _, violin) : violinPointsExtractor; - }; - /** - * Gets or sets the violinPointValueExtractor - * @param {function} [_=none] - * @returns {violin | function} - * @memberof violin - * @property - * by default violinPointsExtractor = function(pointKey, violinKey, violinData, violinValues) {return violinValues[violinKey].values[pointKey]} - */ - violin.violinPointValueExtractor = function (_) { - return arguments.length ? (violinPointValueExtractor = _, violin) : violinPointValueExtractor; - }; - - /** - * Gets or sets the selection in which items are manipulated - * @param {d3.selection} [_=none] - * @returns {violin | d3.selection} - * @memberof violin - * @property - * by default selection = selection - */ - violin.selection = function (_) { - return arguments.length ? (selection = _, violin) : selection; - }; - /** - * Gets or sets the data - * (see {@link violin#data}) - * @param {number} [_=none] - * @returns {violin | object} - * @memberof violin - * @property - */ - violin.data = function (_) { - return arguments.length ? (data = _, violin) : data; - }; - /** - * Gets or sets the orient of the boxes - * (see {@link violin#orient}) - * @param {number} [_=none] - * @returns {violin | object} - * @memberof violin - * @property - */ - violin.orient = function (_) { - return arguments.length ? (orient = _, violin) : orient; - }; - /** - * Gets or sets the amount of horizontal space in which items are manipulated - * (see {@link violin#spaceX}) - * @param {number} [_=none] should be a number > 0 - * @returns {violin | number} - * @memberof violin - * @property - * by default spaceX = undefined - */ - violin.spaceX = function (_) { - return arguments.length ? (spaceX = _, violin) : spaceX; - }; - /** - * Gets or sets the amount of vertical space in which items are manipulated - * (see {@link violin#spaceY}) - * @param {number} [_=none] should be a number > 0 - * @returns {violin | number} - * @memberof violin - * @property - * by default spaceY = undefined - */ - violin.spaceY = function (_) { - return arguments.length ? (spaceY = _, violin) : spaceY; - }; - - /** - * Gets / sets whether or not violin is allowed to go beyond specified dimensions - * (see {@link violin#overflowQ}) - * @param {boolean} [_=none] - * @returns {violin | boolean} - * @memberof violin - * @property - * by default overflowQ = false - */ - violin.overflowQ = function (_) { - return arguments.length ? (overflowQ = _, violin) : overflowQ; - }; - /** - * Gets / sets whether or not to plot points with the violins - * (see {@link violin#pointsQ}) - * @param {boolean} [_=none] - * @returns {violin | boolean} - * @memberof violin - * @property - * by default pointsQ = false - */ - violin.pointsQ = function (_) { - return arguments.length ? (pointsQ = _, violin) : pointsQ; - }; - - /** - * Gets / sets the grouping of the boxes - * (see {@link violin#grouping}) - * @param {Array[]} [_=none] - * @returns {violin | Array[]} - * @memberof violin - * @property - * by default grouping = undefined - */ - violin.grouping = function (_) { - return arguments.length ? (grouping = _, violin) : grouping; - }; - /** - * Gets / sets the valueExtractor - * (see {@link violin#valueExtractor}) - * @param {function} [_=none] - * @returns {violin | function} - * @memberof violin - * @property - */ - violin.valueExtractor = function (_) { - return arguments.length ? (valueExtractor = _, violin) : valueExtractor; - }; - /** - * Gets / sets the sortingFunction - * (see {@link violin#sortingFunction}) - * @param {function} [_=none] - * @returns {violin | function} - * @memberof violin - * @property - */ - violin.sortingFunction = function (_) { - return arguments.length ? (sortingFunction = _, violin) : sortingFunction; - }; - - /** - * Gets / sets the scale for which the violin values should be transformed by - * (see {@link violin#scale}) - * @param {d3.scale} [_=none] - * @returns {violin | d3.scale} - * @memberof violin - * @property - * by default scale = d3.scaleLinear() - */ - violin.scale = function (_) { - return arguments.length ? (scale = _, violin) : scale; - }; - /** - * Gets / sets the padding for the domain of the scale - * (see {@link violin#domainPadding}) - * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin - * @property - * by default domainPadding = 0.5 - */ - violin.domainPadding = function (_) { - return arguments.length ? (domainPadding = _, violin) : domainPadding; - }; - - /** - * Gets / sets objectSpacer - * (see {@link violin#objectSpacer}) - * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin - * @property - * by default objectSpacer = 0.05 - */ - violin.objectSpacer = function (_) { - return arguments.length ? (objectSpacer = _, violin) : objectSpacer; - }; - /** - * Gets / sets the minObjectSize - * (see {@link violin#minObjectSize}) - * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin - * @property - * by default minObjectSize = 15 - */ - violin.minObjectSize = function (_) { - return arguments.length ? (minObjectSize = _, violin) : minObjectSize; - }; - /** - * Gets / sets the maxObjectSize - * (see {@link violin#maxObjectSize}) - * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin - * @property - * by default maxObjectSize = 50 - */ - violin.maxObjectSize = function (_) { - return arguments.length ? (maxObjectSize = _, violin) : maxObjectSize; - }; - - /** - * Gets / sets the objectStrokeWidth - * (see {@link violin#objectStrokeWidth}) - * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin - * @property - * by default objectStrokeWidth = 2 - */ - violin.objectStrokeWidth = function (_) { - return arguments.length ? (objectStrokeWidth = _, violin) : objectStrokeWidth; - }; - - /** - * Gets / sets the colorFunction - * (see {@link violin#colorFunction}) - * @param {colorFunction} [_=none] - * @returns {violin | colorFunction} - * @memberof violin - * @property - * by default colorFunction = colorFunction() - */ - violin.colorFunction = function (_) { - return arguments.length ? (colorFunction$$1 = _, violin) : colorFunction$$1; - }; - /** - * Gets / sets the colorFunction - * (see {@link violin#colorFunction}) - * @param {colorFunction} [_=none] - * @returns {violin | colorFunction} - * @memberof violin - * @property - * by default colorFunction = colorFunction() - */ - violin.pointColorFunc = function (_) { - return arguments.length ? (pointColorFunc = _, violin) : pointColorFunc; - }; - - /** - * Gets / sets the pointRadius - * (see {@link violin#pointRadius}) - * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin - * @property - * by default pointRadius = 2 - */ - violin.pointRadius = function (_) { - return arguments.length ? (pointRadius = _, violin) : pointRadius; - }; - /** - * Gets / sets the pointStrokeWidth - * (see {@link violin#pointStrokeWidth}) - * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin - * @property - * by default pointStrokeWidth = 2 - */ - violin.pointStrokeWidth = function (_) { - return arguments.length ? (pointStrokeWidth = _, violin) : pointStrokeWidth; - }; - - /** - * Gets / sets the backgroundFill - * (see {@link violin#backgroundFill}) - * @param {string} [_=none] - * @returns {violin | string} - * @memberof violin - * @property - * by default backgroundFill = 'transparent' - */ - violin.backgroundFill = function (_) { - return arguments.length ? (backgroundFill = _, violin) : backgroundFill; - }; - /** - * Gets / sets the namespace - * (see {@link violin#namespace}) - * @param {string} [_=none] - * @returns {violin | string} - * @memberof violin - * @property - * by default namespace = 'd3sm-violin' - */ - violin.namespace = function (_) { - return arguments.length ? (namespace = _, violin) : namespace; - }; - /** - * Gets / sets the objectClass - * (see {@link violin#objectClass}) - * @param {string} [_=none] - * @returns {violin | string} - * @memberof violin - * @property - * by default objectClass = 'tick-group' - */ - violin.objectClass = function (_) { - return arguments.length ? (objectClass = _, violin) : objectClass; - }; - - /** - * Gets / sets the transitionDuration - * (see {@link violin#transitionDuration}) - * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin - * @property - * by default transitionDuration = 1000 - */ - violin.transitionDuration = function (_) { - return arguments.length ? (transitionDuration = _, violin) : transitionDuration; - }; - /** - * Gets / sets the easeFunc - * (see {@link violin#easeFunc}) - * @param {d3.ease} [_=none] - * @returns {violin | d3.ease} - * @memberof violin - * @property - * by default easeFunc = d3.easeExp - */ - violin.easeFunc = function (_) { - return arguments.length ? (easeFunc = _, violin) : easeFunc; - }; - - /** - * Gets / sets the quartileKey - * (see {@link violin#quartileKey}) - * @param {string} [_=none] - * @returns {violin | string} - * @memberof violin - * @property - * by default quartileKey = "quartiles" - */ - violin.quartileKey = function (_) { - return arguments.length ? (quartileKey = _, violin) : quartileKey; - }; - /** - * Gets / sets the quartileKeys - * (see {@link violin#quartileKeys}) - * @param {string[]} [_=none] - * @returns {violin | string[]} - * @memberof violin - * @property - * by default quartileKeys = ["Q0","Q1","Q2","Q3","Q4"] - */ - violin.quartileKeys = function (_) { - return arguments.length ? (quartileKeys = _, violin) : quartileKeys; - }; - - /** - * Gets / sets the violinKeys - * (see {@link violin#violinKeys}) - * @param {string[]} [_=none] - * @returns {violin | string[]} - * @memberof violin - * @property - * by default violinKeys = undefined - */ - violin.violinKeys = function (_) { - return arguments.length ? (violinKeys = _, violin) : violinKeys; - }; - /** - * Gets / sets the violinValues - * (see {@link violin#violinValues}) - * @param {Object[]} [_=none] - * @returns {violin | Object[]} - * @memberof violin - * @property - * by default violinValues = undefined - */ - violin.violinValues = function (_) { - return arguments.length ? (violinValues = _, violin) : violinValues; - }; - - /** - * Gets / sets the objectSize - * (see {@link violin#objectSize}) - * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin - * @property - * by default objectSize = undefined - */ - violin.objectSize = function (_) { - return arguments.length ? (objectSize = _, violin) : objectSize; - }; - /** - * Gets / sets the spacerSize - * (see {@link violin#spacerSize}) - * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin - * @property - * by default spacerSize = undefined - */ - violin.spacerSize = function (_) { - return arguments.length ? (spacerSize = _, violin) : spacerSize; - }; - /** - * Gets / sets the tooltip - * (see {@link violin#tooltip}) - * @param {tooltip} [_=none] - * @returns {violin | tooltip} - * @memberof violin - * @property - * by default tooltip = tooltip() - */ - violin.tooltip = function (_) { - return arguments.length ? (tooltip$$1 = _, violin) : tooltip$$1; - }; - /** - * Gets / sets the pointsTooltip - * (see {@link violin#pointsTooltip}) - * @param {tooltip} [_=none] - * @returns {violin | tooltip} - * @memberof violin - * @property - * by default pointsTooltip = tooltip() - */ - violin.pointsTooltip = function (_) { - return arguments.length ? (pointsTooltip = _, violin) : pointsTooltip; - }; - - function violin() { - var _ref, _ref2, _ref3; - - // for convenience in handling orientation specific values - var horizontalQ = orient == 'horizontal' ? true : false; - - // background cliping rectangle - var bgcpRect = { x: 0, y: 0, width: spaceX, height: spaceY }; - var container = setupContainer(selection, namespace, bgcpRect, backgroundFill); - - // if grouping is undefined sort violinKeys by sortingFunction - var ordered = grouping == undefined ? d3.keys(data).sort(sortingFunction) : grouping; - - // console.log(ordered) - - violinKeys = flatten(ordered); - - var calcValues = neededViolinValues().horizontalQ(horizontalQ).quartileKeys(quartileKeys).violinPointsExtractor(violinPointsExtractor).violinPointValueExtractor(violinPointValueExtractor); - - // augment valus - violinKeys.map(function (vk, i) { - calcValues(vk, data); - }); - - var numberOfObjects = violinKeys.length; - - var min = (_ref = []).concat.apply(_ref, toConsumableArray(violinKeys.map(function (k, i) { - return data[k].quartiles[quartileKeys[0]]; - }))); - var max = (_ref2 = []).concat.apply(_ref2, toConsumableArray(violinKeys.map(function (k, i) { - return data[k].quartiles[quartileKeys[quartileKeys.length - 1]]; - }))); - var extent = [Math.min.apply(Math, toConsumableArray(min)) - domainPadding, Math.max.apply(Math, toConsumableArray(max)) + domainPadding]; - // console.log(extent, violinValues, ordered) - - // set the scale - scale.domain(extent).range(horizontalQ ? [0, spaceY] : [0, spaceX]); - var space = horizontalQ ? spaceX : spaceY; - // calculate object size - objectSize = calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ); - // calculate spacer size if needed - spacerSize = calculateWidthOfSpacer(ordered, space, objectSize, numberOfObjects, objectSpacer, overflowQ); - // make the nested groups - var spacerFunction = groupingSpacer().horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects).objectClass(objectClass).objectSize(objectSize).spacerSize(spacerSize).transitionDuration(transitionDuration).easeFunc(easeFunc).namespace(namespace); - - // move stuff - spacerFunction(container, ordered, 0); - // console.log(violinKeys, ordered, container.selectAll('g:not(.to-remove).'+objectClass).nodes()) - - // for color function - var parentIndexArray = []; - container.selectAll('g:not(.to-remove).' + objectClass).each(function (d, i) { - if (hasQ(violinKeys, d)) { - parentIndexArray.push(Number(d3.select(this).attr('parent-index'))); - } - }); - - // update color function - colorFunction$$1 = colorFunction$$1.colorBy() == 'index' ? colorFunction$$1.dataExtent([0, Math.max.apply(Math, parentIndexArray)]) : colorFunction$$1.dataExtent(extent); - - /* violiin specific needs */ - - var frequencyMax = Math.max.apply(Math, toConsumableArray((_ref3 = []).concat.apply(_ref3, toConsumableArray(violinKeys.map(function (k, i) { - return d3.max(data[k].frequencies); - }))))); - var vScale = d3.scaleLinear().domain([0, frequencyMax]).range([0, objectSize / 2]); - - var lArea = d3.line().x(function (d, i) { - return horizontalQ ? -vScale(d.x) : scale(d.x); - }).y(function (d, i) { - return horizontalQ ? scale(extent[1]) - scale(d.y) : -vScale(d.y); - }).curve(d3.curveBasis); - var rArea = d3.line().x(function (d, i) { - return horizontalQ ? vScale(d.x) : scale(d.x); - }).y(function (d, i) { - return horizontalQ ? scale(extent[1]) - scale(d.y) : vScale(d.y); - }).curve(d3.curveBasis); - - container.selectAll('g:not(.to-remove).' + objectClass).each(function (key, i) { - var t = d3.select(this), - currentData = data[key]; - // needed because bug in selecting .to-remove - if (!hasQ(violinKeys, key)) { - return; - } - var i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'), - fillColor = colorFunction$$1(key, currentData, i, 'fill'), - // prevent duplicate computation - strokeColor = colorFunction$$1(key, currentData, i, 'stroke'), - area = safeSelect(t, 'g', 'area'), - la = safeSelect(area, 'path', 'left'), - ra = safeSelect(area, 'path', 'right'), - quarts = safeSelect(t, 'g', 'quarts'), - lq3 = safeSelect(quarts, 'line', 'q3'), - lq1 = safeSelect(quarts, 'line', 'q1'), - q3 = currentData.quartiles[quartileKeys[3]], - q2 = currentData.quartiles[quartileKeys[2]], - q1 = currentData.quartiles[quartileKeys[1]]; - - t.attr('transform', horizontalQ ? 'translate(' + objectSize / 2 + ',0)' : 'translate(0,' + objectSize / 2 + ')'); - // draw curve - la.transition().duration(transitionDuration).attr('d', function (dd, ii) { - return lArea(currentData.contour); - }).attr('fill', fillColor).attr('stroke', strokeColor).attr('stroke-width', objectStrokeWidth); - - ra.transition().duration(transitionDuration).attr('d', function (dd, ii) { - return rArea(currentData.contour); - }).attr('fill', fillColor).attr('stroke', strokeColor).attr('stroke-width', objectStrokeWidth); - - area.node().addEventListener('mouseover', function (dd, ii) { - container.selectAll('g.' + objectClass).style('opacity', 0.2); - t.style('opacity', 1); - la.attr('stroke-width', objectStrokeWidth * 2); - ra.attr('stroke-width', objectStrokeWidth * 2); - }); - area.node().addEventListener('mouseout', function (dd, ii) { - container.selectAll('g.' + objectClass).style('opacity', 1); - la.attr('stroke-width', objectStrokeWidth); - ra.attr('stroke-width', objectStrokeWidth); - }); - - if (pointsQ) { - var ptsContainer = safeSelect(t, 'g', 'points'); - var pts = ptsContainer.selectAll('.point').data(currentData.pointKeys); - pts.on('mouseover', null); - - var ptsExit = pts.exit().transition().ease(easeFunc).duration(transitionDuration).attr('r', 0).attr('cy', horizontalQ ? scale(extent[1]) - scale(q2) : vScale(0)).attr('cx', horizontalQ ? vScale(0) : scale(q2)).remove(); - - var ptsEnter = pts.enter().append('circle').attr('class', 'point').attr('r', 0).attr('cx', horizontalQ ? 0 : scale(q2)).attr('cy', horizontalQ ? scale(q2) : 0); - - pts = pts.merge(ptsEnter); - - // console.log(pointsTooltip.header()) - - var pTTips = tooltip().selection(pts).data(violinPointsExtractor(key, currentData)).header(pointsTooltip.header()).keys(pointsTooltip.keys()).values(pointsTooltip.values()); - - pTTips(); - - var pMin = d3.min(currentData.pointValues), - pMax = d3.max(currentData.pointValues); - - pts.transition().duration(transitionDuration).ease(easeFunc).attr('r', pointRadius).attr('cy', function (pointKey, ii) { - var dd = currentData.pointValues[ii]; - if (horizontalQ) { - return scale(extent[1]) - scale(dd); - } - var j = whichBin(currentData.binned, dd); - var r = Math.random(); - var n = vScale(r * currentData.frequencies[j] * 0.5); - var k = Math.random() > 0.5 ? n : -n; - return k; - }).attr('cx', function (pointKey, ii) { - var dd = currentData.pointValues[ii]; - if (horizontalQ) { - var j = whichBin(currentData.binned, dd); - var r = Math.random(); - var n = vScale(r * currentData.frequencies[j] * 0.5); - var k = Math.random() > 0.5 ? n : -n; - return k; - } - return scale(dd); - }).attr('stroke', function (dd, ii) { - var dd = currentData.pointValues[ii];return pointColorFunc(dd, 'stroke', strokeColor, pMin, pMax); - }).attr('fill', function (dd, ii) { - var dd = currentData.pointValues[ii];return pointColorFunc(dd, 'fill', strokeColor, pMin, pMax); - }).attr('stroke-width', pointStrokeWidth); - - ptsContainer.selectAll('circle.point').on('mouseover', function (dd, ii) { - container.selectAll('g.' + objectClass).style('opacity', 0.2); - t.style('opacity', 1); - la.attr('stroke-width', objectStrokeWidth * 2); - ra.attr('stroke-width', objectStrokeWidth * 2); - - container.selectAll('.point').style('opacity', 0.2); - d3.select(this).style('opacity', 1).attr('r', pointRadius * 2).attr('stroke-width', pointStrokeWidth * 2); - }); - ptsContainer.selectAll('circle.point').on('mouseout', function (dd, ii) { - var e = document.createEvent('SVGEvents'); - e.initEvent('mouseout', true, true); - area.node().dispatchEvent(e); - - container.selectAll('.point').style('opacity', 1); - d3.select(this).attr('stroke-width', pointStrokeWidth).attr('r', pointRadius); - }); - } else { - cV.selectAll('.point').transition().duration(transitionDuration).ease(easeFunc).attr('r', 0).attr('cy', horizontalQ ? scale(extent[1]) - scale(q2) : vScale(0)).attr('cx', horizontalQ ? vScale(0) : scale(q2)).remove(); - } - }); - - tooltip$$1.selection(container.selectAll('g:not(.to-remove).' + objectClass + ' .area')); - if (tooltip$$1.data() == undefined) { - tooltip$$1.data(data); - } - tooltip$$1(); - if (tooltip$$1.values() == undefined) { - tooltip$$1.values([function (currentData, tooltipKey) { - return currentData['quartiles'][tooltipKey]; - }, function (currentData, tooltipKey) { - return currentData['quartiles'][tooltipKey]; - }, function (currentData, tooltipKey) { - return currentData['quartiles'][tooltipKey]; - }, function (currentData, tooltipKey) { - return currentData['quartiles'][tooltipKey]; - }, function (currentData, tooltipKey) { - return currentData['quartiles'][tooltipKey]; - }]); - } - } - - return violin; - } - - /** - * Produces the function which manipulates the violin data to have values needed - * for rendering the violins as svg. - * @returns {function} calculateViolinValues - * @namespace neededViolinValues - */ - function neededViolinValues() { - var - /** - * Whether or not the orientation of the violins are horizontal - * (see {@link violin#orient}) - * @param {Object} [horizontalQ=true] - * @memberof neededViolinValues# - * @property - */ - horizontalQ = true, - - /** - * Keys to be put into the quartiles if they need to be calculated. - * (see {@link violin#quartileKeys}) - * @param {Object} [quartileKeys=['Q0', 'Q1', 'Q2', 'Q3', 'Q4']] - * @memberof neededViolinValues# - * @property - */ - quartileKeys = ['Q0', 'Q1', 'Q2', 'Q3', 'Q4'], - - /** - * Function which given the key of the violin and that key's associated value - * returns the object consiting of the points of the violin - * (see {@link violin#violinPointsExtractor}) - * @param {Object} [violinPointsExtractor=function(violinKey, violinData) { return violinData.points }] - * @memberof neededViolinValues# - * @property - */ - violinPointsExtractor, - - /** - * Function which given the key of the current point and the object of points for the - * violin, returns the numerical value of the point - * (see {@link violin#violinPointValueExtractor}) - * @param {Object} [violinPointValueExtractor=function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value }] - * @memberof neededViolinValues# - * @property - */ - violinPointValueExtractor; - - /** - * Gets / sets the horizontalQ - * (see {@link violin#orient}) - * @param {boolean} [_=none] - * @returns {calculateViolinValues | boolean} - * @memberof calculateViolinValues - * @property - * - * by default horizontalQ = true - */ - calculateViolinValues.horizontalQ = function (_) { - return arguments.length ? (horizontalQ = _, calculateViolinValues) : horizontalQ; - }; - /** - * Gets / sets the quartileKeys - * (see {@link violin#quartileKeys}) - * @param {string[]} [_=none] - * @returns {calculateViolinValues | string[]} - * @memberof calculateViolinValues - * @property - * - * by default quartileKeys = ["Q0","Q1","Q2","Q3","Q4"] - */ - calculateViolinValues.quartileKeys = function (_) { - return arguments.length ? (quartileKeys = _, calculateViolinValues) : quartileKeys; - }; - /** - * Gets / sets the violinPointsExtractor - * (see {@link violin#violinPointsExtractor}) - * @param {function} [_=none] - * @returns {calculateViolinValues | function} - * @memberof calculateViolinValues - * @property - * - * by default violinPointsExtractor = function(violinKey, violinData) { return violinData.points } - */ - calculateViolinValues.violinPointsExtractor = function (_) { - return arguments.length ? (violinPointsExtractor = _, calculateViolinValues) : violinPointsExtractor; - }; - /** - * Gets / sets the violinPointValueExtractor - * (see {@link violin#violinPointValueExtractor}) - * @param {function} [_=none] - * @returns {calculateViolinValues | function} - * @memberof calculateViolinValues - * @property - * - * by default violinPointValueExtractor = function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value } - */ - calculateViolinValues.violinPointValueExtractor = function (_) { - return arguments.length ? (violinPointValueExtractor = _, calculateViolinValues) : violinPointValueExtractor; - }; - - /** - * The function produced by neededViolinValues. - * - * Adds the data need to render the violin as an svg - * @param {string} violinKey the key of the current violin - * @param {object} data the object consisting of violinKey - values (violin data) pairs - * @returns {none} this function manipulates the passed data object adding the following keys - * - * data[violinKey].binned // the binned values of the points - * - * data[violinKey].frequencies // the list consisting of the length of each bin - * - * data[violinKey].contour // the points depicting the contour of 1/2 of the violin - * - * data[violinKey].quartiles // the quartiles of the points' values - * - * data[violinKey].pointKeys // the keys associated with the points - * - * data[violinKey].pointValues // the numerical values of the points - * - * @memberof neededViolinValues# - * @property - */ - function calculateViolinValues(violinKey, data) { - // data for the current violin - var violinData = data[violinKey]; - // the object of points - var violinPoints = violinPointsExtractor(violinKey, violinData); - // - var violinPointsKeys = d3.keys(violinPoints); - // the numerical values of those points - var violinPointsValues = violinPointsKeys.map(function (pk, i) { - return violinPointValueExtractor(pk, violinPoints); - }); - - // quartiles of those points - var pointQuartiles = quartiles(violinPointsValues, quartileKeys); - - // binned points - var binned = d3.histogram()(violinPointsValues); - // length of bins - var frequencies = binned.map(function (bin) { - return bin.length; - }); - // min and max countour points for nice drawings - var minContourPoint = horizontalQ ? { x: 0, y: d3.min(violinPointsValues) } : { x: d3.min(violinPointsValues), y: 0 }; - var maxContourPoint = horizontalQ ? { x: 0, y: d3.max(violinPointsValues) } : { x: d3.max(violinPointsValues), y: 0 }; - var violinContourPoints = binned.map(function (bin, i) { - return horizontalQ ? { y: bin.length ? d3.median(bin) : d3.median([bin.x0, bin.x1]), x: frequencies[i] } : { x: bin.length ? d3.median(bin) : d3.median([bin.x0, bin.x1]), y: frequencies[i] }; - }); - // points along which to draw the violin shpe - violinContourPoints = [minContourPoint].concat(violinContourPoints).concat([maxContourPoint]); - - // set data - violinData.binned = binned; - violinData.frequencies = frequencies; - violinData.contour = violinContourPoints; - violinData.quartiles = pointQuartiles; - violinData.pointKeys = violinPointsKeys; - violinData.pointValues = violinPointsValues; - } - - return calculateViolinValues; - } - - function numericLegend(selection) { - - var min = 0, - max = 1, - spaceX, - spaceY, - colorFunction$$1 = colorFunction(), - namespace = 'd3sm-linear-vertical-gradient', - fontSize = 12, - backgroundFill = 'transparent', - textColor = 'black', - roundTo = 2; - - legend.min = function (_) { - return arguments.length ? (min = _, legend) : min; - }; - legend.max = function (_) { - return arguments.length ? (max = _, legend) : max; - }; - legend.spaceX = function (_) { - return arguments.length ? (spaceX = _, legend) : spaceX; - }; - legend.spaceY = function (_) { - return arguments.length ? (spaceY = _, legend) : spaceY; - }; - legend.namespace = function (_) { - return arguments.length ? (namespace = _, legend) : namespace; - }; - legend.fontSize = function (_) { - return arguments.length ? (fontSize = _, legend) : fontSize; - }; - legend.backgroundFill = function (_) { - return arguments.length ? (backgroundFill = _, legend) : backgroundFill; - }; - legend.colorFunction = function (_) { - return arguments.length ? (colorFunction$$1 = _, legend) : colorFunction$$1; - }; - legend.textColor = function (_) { - return arguments.length ? (textColor = _, legend) : textColor; - }; - legend.roundTo = function (_) { - return arguments.length ? (roundTo = _, legend) : roundTo; - }; - - function legend() { - // background cliping rectangle - var bgcpRect = { x: 0, y: 0, width: spaceX, height: spaceY }; - var container = setupContainer(selection, namespace, bgcpRect, backgroundFill); - - var defs = safeSelect(selection, 'defs'); - var linearGradient = safeSelect(defs, 'linearGradient').attr("x1", "0%").attr("y1", "100%").attr("x2", "0%").attr("y2", "0%").attr('id', hypenate(namespace, 'numerical-legend-gradient')); - - colorFunction$$1.dataExtent([min, max]).colorBy('value').valueExtractor(function (k, v, i) { - return v; - }); - - linearGradient.selectAll('stop').data(colorFunction$$1.colors()).enter().append('stop').attr("offset", function (d, i) { - return i / (colorFunction$$1.colors().length - 1); - }).attr('stop-color', function (d) { - return d; - }); - - var rect = safeSelect(container, 'rect', 'legend').attr('transform', 'translate(0,' + fontSize + ')').style("fill", "url(#" + hypenate(namespace, 'numerical-legend-gradient') + ")").attr('x', 0).attr('y', 0).attr('width', spaceX).attr('height', spaceY - fontSize * 2).on('mousemove', function (d, i) { - legendMousemove(d, i, rect, d3.select(this)); - }).on('mouseout', function (d, i) { - d3.select("#" + hypenate(namespace, 'legend-tooltip')).remove(); - }); - - var minText = safeSelect(container, 'text', 'min').text(round(min, 2)).attr('text-anchor', 'middle').attr("font-size", fontSize + 'px').attr('transform', function (d, i) { - var x = spaceX / 2, - y = spaceY - fontSize / 4, - t = 'translate(' + x + ',' + y + ')'; - return t; - }); - - var maxText = safeSelect(container, 'text', 'max').text(round(max, 2)).attr('text-anchor', 'middle').attr("font-size", fontSize + 'px').attr('transform', function (d, i) { - var x = spaceX / 2, - y = fontSize, - t = 'translate(' + x + ',' + y + ')'; - return t; - }); - } - - function legendMousemove(d, i, rect, t) { - var s = d3.scaleLinear().domain([0, rect.attr('height')]).range([max, min]); - var m = d3.mouse(rect.node()); - var v = round(s(m[1]), roundTo); - - var strokeColor = colorFunction$$1(undefined, v, undefined, 'stroke'); - var fillColor = colorFunction$$1(undefined, v, undefined, 'fill'); - - var div = safeSelect(d3.select('body'), 'div', hypenate(namespace, 'legend-tooltip')).attr('id', hypenate(namespace, 'legend-tooltip')).style('position', 'absolute').style('left', d3.event.pageX + 15 + 'px').style('top', d3.event.pageY + 15 + 'px').style('background-color', fillColor).style('border-color', strokeColor).style('min-width', fontSize * (String(max).split('.')[0].length + 3) + 'px').style('min-height', fontSize * (String(max).split('.')[0].length + 3) + 'px').style('border-radius', '50%').style('border-radius', '5000px').style('display', 'flex').style('justify-content', 'center').style('text-align', 'middle').style('padding', 2 + "px").style('border-style', 'solid').style('border-width', 2); - - var text = safeSelect(div, 'div').text(v).style('color', textColor).style('align-self', 'center'); - } - - return legend; - } - - function categoricLegend(selection) { - var categories, - - - /** - * Data to plot. Assumed to be a object, where each key corresponds to a legend - * (see {@link legend#data}) - * @param {Object} [data=undefined] - * @memberof legend# - * @property - */ - data, - - /** - * Which direction to render the bars in - * (see {@link legend#orient}) - * @param {number} [orient='horizontal'] - * @memberof legend# - * @property - */ - orient = 'horizontal', - - /** - * Amount of horizontal space (in pixels) avaible to render the legend in - * (see {@link legend#spaceX}) - * @param {number} [spaceX=undefined] - * @memberof legend# - * @property - */ - spaceX, - - /** - * Amount of vertical space (in pixels) avaible to render the legend in - * (see {@link legend.spaceY}) - * @param {number} [spaceY=undefined] - * @memberof legend# - * @property - */ - spaceY, - - - /** - * Whether or not to allow legend to render elements pass the main spatial dimension - * given the orientation (see {@link legend#orient}), where {@link legend#orient}="horizontal" - * the main dimension is {@link legend#spaceX} and where {@link legend#orient}="vertical" - * the main dimension is {@link legend#spaceY} - * @param {boolean} [overflowQ=false] - * @memberof legend# - * @property - */ - overflowQ = false, - - - /** - * An array - putatively of other arrays - depicting how bars should be arranged - * @param {Array[]} [grouping=undefined] - * @memberof legend# - * @property - */ - grouping, - - - /** - * How to get the value of the legend - * @param {function} [valueExtractor=function(key, index) { return data[key] }] - * @memberof legend# - * @property - */ - valueExtractor = function valueExtractor(key, index) { - return data[key]; - }, - - /** - * How to sort the bars - if {@link bar#grouping} is not provided. - * @param {function} [sortingFunction=function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}] - * @memberof bar# - * @property - */ - sortingFunction = function sortingFunction(keyA, keyB) { - return d3.ascending(keyA, keyB); - }, - - /** - * Default space for the spacer (percentage) of main dimension given the orientation - * (see {@link legend#orient}), where {@link legend#orient}="horizontal" - * the main dimension is {@link legend#spaceX} and where {@link legend#orient}="vertical" - * the main dimension is {@link legend#spaceY} between bars - * @param {number} [objectSpacer=0.05] - * @memberof legend# - * @property - */ - objectSpacer = 0.05, - - /** - * The minimum size that an object can be - * @param {number} [minObjectSize=50] - * @memberof legend# - * @property - */ - minObjectSize = 10, - - /** - * The maximum size that an object can be - * @param {number} [maxObjectSize=100] - * @memberof legend# - * @property - */ - maxObjectSize = 100, - - - /** - * The stroke width of the bars - * @param {number} [barStrokeWidth=2] - * @memberof legend# - * @property - */ - bubbleStrokeWidth = 2, - - /** - * Instance of ColorFunction - * @param {function} [colorFunction = colorFunction()] - * @memberof legend# - * @property - */ - colorFunction$$1 = colorFunction(), - - - /** - * Color of the background - * @param {string} [backgroundFill="transparent"] - * @memberof legend# - * @property - */ - backgroundFill = 'transparent', - - /** - * Namespace for all items made by this instance of legend - * @param {string} [namespace="d3sm-legend"] - * @memberof legend# - * @property - */ - namespace = 'd3sm-legend', - - /** - * Class name for legend container ( element) - * @param {string} [objectClass="legend"] - * @memberof legend# - * @property - */ - objectClass = 'legend', - - - /** - * Duration of all transitions of this element - * @param {number} [transitionDuration=1000] - * @memberof legend# - * @property - */ - transitionDuration = 1000, - - /** - * Easing function for transitions - * @param {d3.ease} [easeFunc=d3.easeExp] - * @memberof legend# - * @property - */ - easeFunc = d3.easeExp; - - legend.categories = function (_) { - return arguments.length ? (categories = _, legend) : categories; - }; - - /** - * Gets or sets the selection in which items are manipulated - * @param {d3.selection} [_=none] - * @returns {legend | d3.selection} - * @memberof legend - * @property - * by default selection = selection - */ - legend.selection = function (_) { - return arguments.length ? (selection = _, legend) : selection; - }; - /** - * Gets or sets the data - * (see {@link legend#data}) - * @param {number} [_=none] - * @returns {legend | object} - * @memberof legend - * @property - */ - legend.data = function (_) { - return arguments.length ? (data = _, legend) : data; - }; - /** - * Gets or sets the orient of the bars - * (see {@link legend#orient}) - * @param {number} [_=none] - * @returns {legend | object} - * @memberof legend - * @property - */ - legend.orient = function (_) { - return arguments.length ? (orient = _, legend) : orient; - }; - /** - * Gets or sets the amount of horizontal space in which items are manipulated - * (see {@link legend#spaceX}) - * @param {number} [_=none] should be a number > 0 - * @returns {legend | number} - * @memberof legend - * @property - * by default spaceX = undefined - */ - legend.spaceX = function (_) { - return arguments.length ? (spaceX = _, legend) : spaceX; - }; - /** - * Gets or sets the amount of vertical space in which items are manipulated - * (see {@link legend#spaceY}) - * @param {number} [_=none] should be a number > 0 - * @returns {legend | number} - * @memberof legend - * @property - * by default spaceY = undefined - */ - legend.spaceY = function (_) { - return arguments.length ? (spaceY = _, legend) : spaceY; - }; - - /** - * Gets / sets whether or not legend is allowed to go beyond specified dimensions - * (see {@link legend#spaceX}) - * @param {boolean} [_=none] - * @returns {legend | boolean} - * @memberof legend - * @property - * by default overflowQ = false - */ - legend.overflowQ = function (_) { - return arguments.length ? (overflowQ = _, legend) : overflowQ; - }; - /** - * Gets / sets the grouping of the bars - * (see {@link legend#grouping}) - * @param {Array[]} [_=none] - * @returns {legend | Array[]} - * @memberof legend - * @property - * by default grouping = undefined - */ - legend.grouping = function (_) { - return arguments.length ? (grouping = _, legend) : grouping; - }; - /** - * Gets / sets the valueExtractor - * (see {@link legend#valueExtractor}) - * @param {function} [_=none] - * @returns {legend | function} - * @memberof legend - * @property - * by default valueExtractor = function(key, index) { return data[key] }, - */ - legend.valueExtractor = function (_) { - return arguments.length ? (valueExtractor = _, legend) : valueExtractor; - }; - /** - * Gets / sets the sortingFunction - * (see {@link bar#sortingFunction}) - * @param {function} [_=none] - * @returns {bar | function} - * @memberof bar - * @property - * by default sortingFunction = function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}, - */ - legend.sortingFunction = function (_) { - return arguments.length ? (sortingFunction = _, legend) : sortingFunction; - }; - /** - * Gets / sets objectSpacer - * (see {@link legend#objectSpacer}) - * @param {number} [_=none] - * @returns {legend | number} - * @memberof legend - * @property - * by default objectSpacer = 0.05 - */ - legend.objectSpacer = function (_) { - return arguments.length ? (objectSpacer = _, legend) : objectSpacer; - }; - /** - * Gets / sets the minObjectSize - * (see {@link legend#minObjectSize}) - * @param {number} [_=none] - * @returns {legend | number} - * @memberof legend - * @property - * by default minObjectSize = 50 - */ - legend.minObjectSize = function (_) { - return arguments.length ? (minObjectSize = _, legend) : minObjectSize; - }; - /** - * Gets / sets the maxObjectSize - * (see {@link legend#maxObjectSize}) - * @param {number} [_=none] - * @returns {legend | number} - * @memberof legend - * @property - * by default maxObjectSize = 100 - */ - legend.maxObjectSize = function (_) { - return arguments.length ? (maxObjectSize = _, legend) : maxObjectSize; - }; - - /** - * Gets / sets the barStrokeWidth - * (see {@link legend#barStrokeWidth}) - * @param {number} [_=none] - * @returns {legend | number} - * @memberof legend - * @property - * by default barStrokeWidth = 2 - */ - legend.bubbleStrokeWidth = function (_) { - return arguments.length ? (bubbleStrokeWidth = _, legend) : bubbleStrokeWidth; - }; - /** - * Gets / sets the colorFunction - * (see {@link legend#colorFunction}) - * @param {number} [_=none] - * @returns {legend | number} - * @memberof legend - * @property - * by default colorFunction = colorFunction() - */ - legend.colorFunction = function (_) { - return arguments.length ? (colorFunction$$1 = _, legend) : colorFunction$$1; - }; - - /** - * Gets / sets the backgroundFill - * (see {@link legend#backgroundFill}) - * @param {string} [_=none] - * @returns {legend | string} - * @memberof legend - * @property - * by default backgroundFill = 'transparent' - */ - legend.backgroundFill = function (_) { - return arguments.length ? (backgroundFill = _, legend) : backgroundFill; - }; - /** - * Gets / sets the namespace - * (see {@link legend#namespace}) - * @param {string} [_=none] - * @returns {legend | string} - * @memberof legend - * @property - * by default namespace = 'd3sm-legend' - */ - legend.namespace = function (_) { - return arguments.length ? (namespace = _, legend) : namespace; - }; - /** - * Gets / sets the objectClass - * (see {@link legend#objectClass}) - * @param {string} [_=none] - * @returns {legend | string} - * @memberof legend - * @property - * by default objectClass = 'tick-group' - */ - legend.objectClass = function (_) { - return arguments.length ? (objectClass = _, legend) : objectClass; - }; - /** - * Gets / sets the transitionDuration - * (see {@link legend#transitionDuration}) - * @param {number} [_=none] - * @returns {legend | number} - * @memberof legend - * @property - * by default transitionDuration = 1000 - */ - legend.transitionDuration = function (_) { - return arguments.length ? (transitionDuration = _, legend) : transitionDuration; - }; - /** - * Gets / sets the easeFunc - * (see {@link legend#easeFunc}) - * @param {d3.ease} [_=none] - * @returns {legend | d3.ease} - * @memberof legend - * @property - * by default easeFunc = d3.easeExp - */ - legend.easeFunc = function (_) { - return arguments.length ? (easeFunc = _, legend) : easeFunc; - }; - - function legend() { - var horizontalQ = orient == 'horizontal' ? true : false; - var verticalQ = !horizontalQ; - // background cliping rectangle - var bgcpRect = { x: 0, y: 0, width: spaceX, height: spaceY }; - var container = setupContainer(selection, namespace, bgcpRect, backgroundFill); - - colorFunction$$1.dataExtent([0, categories.length - 1]).colorBy('categories').categoryExtractor(function (k, v, i) { - return v; - }); - - var r = Math.min(spaceX, spaceY) / 2; - var numberOfObjects = categories.length; - - // if grouping is undefined sort barKeys by sortingFunction - var ordered = grouping == undefined ? categories.sort(sortingFunction) : grouping; - // ordered might be nested depending on grouping - var catKeys = flatten(ordered); - - var space = horizontalQ ? spaceX : spaceY; - // calculate object size - var objectSize = calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ); - // calculate spacer size if needed - var spacerSize = calculateWidthOfSpacer(catKeys, space, objectSize, numberOfObjects, objectSpacer, overflowQ); - // make the nested groups - var spacerFunction = groupingSpacer().horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects).objectClass(objectClass).objectSize(objectSize).spacerSize(spacerSize).transitionDuration(transitionDuration).easeFunc(easeFunc).namespace(namespace); - - spacerFunction(container, ordered, 0); - var r = Math.min(objectSize, spaceX, spaceY) / 2 - bubbleStrokeWidth; - - container.selectAll('g:not(.to-remove).' + objectClass).each(function (cat, i) { - var t = d3.select(this); - var c = safeSelect(t, 'circle'); - var fillColor = colorFunction$$1(undefined, cat, i, 'fill'), - // prevent duplicate computation - strokeColor = colorFunction$$1(undefined, cat, i, 'stroke'); - - var cx = horizontalQ ? r + bubbleStrokeWidth : (spaceX - r * 2) / 2 + r; - var cy = verticalQ ? r + bubbleStrokeWidth : (spaceX - r * 2) / 2 + r; - - c.attr("r", r).attr('cx', cx).attr('cy', cy).attr('fill', fillColor).attr('stroke', strokeColor).attr('stroke-width', bubbleStrokeWidth); - - var text = safeSelect(t, 'text'); - text.text(cat).attr('text-anchor', 'middle').attr("transform", function (d, i) { - var x = cx, - y = cy + text.node().getBoundingClientRect().height / 4, - t = 'translate(' + x + ',' + y + ')'; - return t; - }); - }); - } - - return legend; - } - - /******************************************************************************* - ** ** - ** ** - ** D3 EXTENSIONS ** - ** ** - ** ** - *******************************************************************************/ - /** - * Recursively ascends parents of selection until it finds an svg tag - * @function d3.selection.thisSVG - * @augments d3.selection - * @returns {Element} which is the svg tag, not the d3 selection of that tag - */ - d3.selection.prototype.thisSVG = function () { - return getContainingSVG(this.node()); - }; - - /** - * Helper for getting absolute position of the mouse - * @function d3.mouse.absolute - * @augments d3.mouse - * @returns {number[]} [x, y] as they relate to `html` not to local scope. - */ - d3.mouse.absolute = function () { - var html = d3.select('html').node(); - - var _ref = this(html), - _ref2 = slicedToArray(_ref, 2), - x = _ref2[0], - y = _ref2[1]; - - return [x, y]; - }; - - /** - * Gets position of the selection in relation to the containing svg - * @see{@link getContainingSVG} - * @function d3.selection.absolutePosition - * @augments d3.selection - * @returns {Object} with structure similar to getBoundingClientRect, e.g. - * top, left, bottom, right, height, width - */ - d3.selection.prototype.absolutePosition = function () { - var element = this.node(); - var elementPosition = element.getBoundingClientRect(); - var containerSVG = getContainingSVG(element); - var svgPosition = containerSVG.getBoundingClientRect(); - - return { - top: elementPosition.top - svgPosition.top, - left: elementPosition.left - svgPosition.left, - bottom: elementPosition.bottom - svgPosition.top, - right: elementPosition.right - svgPosition.left, - height: elementPosition.height, - width: elementPosition.width - }; - }; - - d3.selection.prototype.relativePositionTo = function (container) { - var element = this.node(); - var elementPosition = element.getBoundingClientRect(); - var containerSVG = container; - var svgPosition = containerSVG.getBoundingClientRect(); - - return { - top: elementPosition.top - svgPosition.top, - left: elementPosition.left - svgPosition.left, - bottom: elementPosition.bottom - svgPosition.top, - right: elementPosition.right - svgPosition.left, - height: elementPosition.height, - width: elementPosition.width - }; - }; - - function getTranslation$1(selection) { - var transform = selection.attr('transform'); - - var _transform$split = transform.split('translate('), - _transform$split2 = slicedToArray(_transform$split, 2), - junk = _transform$split2[0], - xy = _transform$split2[1]; - - var _xy$split = xy.split(','), - _xy$split2 = slicedToArray(_xy$split, 2), - x = _xy$split2[0], - y = _xy$split2[1]; - - junk = y.split(')'); - return [parseFloat(x), parseFloat(y)]; - } - - function lasso(selection) { - var svg, - // svg that is target of events - objectContainer, - // container which houses objects we are selecting (allows for transform to be applied to lasso) - objectClass, - // class of object we are selecting - namespace = "d3sm-lasso", - chartContainer, - chartOffset, - objectsOffset, - eventCatcher, - xScale, - // optional scale for the lasso currentPoints - yScale, - // optional scale for the lasso currentPoints - - activeQ = false, - // whether or not lasso is active - - currentPoints = [], - // mouse points for current lasso - allPoints = [], - // list of lists for all points of lassos - - line = d3.line().x(function (d, i) { - var x; - if (xScale != undefined) { - x = xScale(d[0]); - } else { - x = d[0]; - } - return x; //- chartOffset[0]// - objectsOffset[0] - }).y(function (d, i) { - var y; - if (yScale != undefined) { - y = yScale(d[1]); - } else { - y = d[1]; - } - return y; // - chartOffset[1]// - objectsOffset[1] - }).curve(d3.curveLinearClosed), - instance = 0, - // an indentifier for which instance this lasso is under the current svg - - tickDistance = 10, - - - // styles for lasso path - color = '#17a2b8', - animationRate = '10s', - opacity = 0.3, - dashArray = '5, 10', - stroke = 'black', - strokeWidth = 2, - - - // styles for lassoed objects - lassoedFill = "white", - lassoedStroke = 'black', - lassoedStrokeWidth = 3, - transitionDuration = 1000, - easeFunc = d3.easeExp; - - var path; - - lasso.svg = function (_) { - return arguments.length ? (svg = _, lasso) : svg; - }; - lasso.chartContainer = function (_) { - return arguments.length ? (chartContainer = _, lasso) : chartContainer; - }; - lasso.objectContainer = function (_) { - return arguments.length ? (objectContainer = _, lasso) : objectContainer; - }; - lasso.objectClass = function (_) { - return arguments.length ? (objectClass = _, lasso) : objectClass; - }; - lasso.namespace = function (_) { - return arguments.length ? (namespace = _, lasso) : namespace; - }; - lasso.xScale = function (_) { - return arguments.length ? (xScale = _, lasso) : xScale; - }; - lasso.yScale = function (_) { - return arguments.length ? (yScale = _, lasso) : yScale; - }; - lasso.activeQ = function (_) { - return arguments.length ? (activeQ = _, lasso) : activeQ; - }; - lasso.currentPoints = function (_) { - return arguments.length ? (currentPoints = _, lasso) : currentPoints; - }; - lasso.allPoints = function (_) { - return arguments.length ? (allPoints = _, lasso) : allPoints; - }; - lasso.instance = function (_) { - return arguments.length ? (instance = _, lasso) : instance; - }; - lasso.tickDistance = function (_) { - return arguments.length ? (tickDistance = _, lasso) : tickDistance; - }; - lasso.color = function (_) { - return arguments.length ? (color = _, lasso) : color; - }; - lasso.animationRate = function (_) { - return arguments.length ? (animationRate = _, lasso) : animationRate; - }; - lasso.opacity = function (_) { - return arguments.length ? (opacity = _, lasso) : opacity; - }; - lasso.dashArray = function (_) { - return arguments.length ? (dashArray = _, lasso) : dashArray; - }; - lasso.stroke = function (_) { - return arguments.length ? (stroke = _, lasso) : stroke; - }; - lasso.lassoedFill = function (_) { - return arguments.length ? (lassoedFill = _, lasso) : lassoedFill; - }; - lasso.lassoedStroke = function (_) { - return arguments.length ? (lassoedStroke = _, lasso) : lassoedStroke; - }; - lasso.lassoedStrokeWidth = function (_) { - return arguments.length ? (lassoedStrokeWidth = _, lasso) : lassoedStrokeWidth; - }; - lasso.eventCatcher = function (_) { - return arguments.length ? (eventCatcher = _, lasso) : eventCatcher; - }; - - lasso.drag = drag; - lasso.draw = draw; - lasso.tick = tick; - lasso.detect = detect; - lasso.toggle = toggle; - lasso.remove = remove; - lasso.render = render; - lasso.keyFrames = keyFrames; - lasso.updateObjects = updateObjects; - lasso.applyPathAttributes = applyPathAttributes; - lasso.applyObjectAttributes = applyObjectAttributes; - - keyFrames(); - - function lasso() { - // add a dash animation if needed - if (activeQ) { - transitionDraw(); - } - } - - function toggle(state) { - // use optional param to set state, otherwise toggle state - activeQ = state != undefined ? state : !activeQ; - chartOffset = getTranslation$1(chartContainer); - objectsOffset = getTranslation$1(objectContainer); - - if (activeQ) { - svg.node().addEventListener('mousedown', render, true); - } else { - svg.node().removeEventListener('mousedown', render, true); - remove(); - } - } - - function draw() { - chartOffset = getTranslation$1(chartContainer); - objectsOffset = getTranslation$1(objectContainer); - - var container = safeSelect(objectContainer, 'g', 'lasso-container'); - var paths = container.selectAll('path[instance="' + instance + '"]'); - - // update - paths = paths.data(allPoints); - - // remove excess - var pExit = paths.exit().remove(); - // add needed paths - var pEnter = paths.enter().append('path'); - - // merge - paths = paths.merge(pEnter); - - // apply - applyPathAttributes(paths); - } - - function remove() { - var container = safeSelect(objectContainer, 'g', 'lasso-container'); - var paths = container.selectAll('path[instance="' + instance + '"]').remove(); - container.remove(); - objectContainer.selectAll(objectClass).classed("in-lasso", false); - updateObjects(); - } - - function render(event) { - // nothing can interefer with drawing the lasso - event.preventDefault();event.stopPropagation(); - - var container = safeSelect(objectContainer, 'g', 'lasso-container'); - - /* - each time the user presses down, while the state is active, the lasso - the lasso should make a seperate segment. - */ - currentPoints = []; - - svg.node().addEventListener('mousemove', drag); - svg.node().addEventListener('mouseup', function (event) { - svg.node().removeEventListener('mousemove', drag); - allPoints.push(currentPoints); - // BUG: somehow this is pushing currentPoints n times where n is the nth lasso path for the current instance - // NOTE: allPoints = unique(allPoints) is a temporary and inefficient fix - allPoints = unique(allPoints); - }); - - path = container.append('path').data([currentPoints]); - applyPathAttributes(path); - } - - function transitionDraw() { - var container = safeSelect(objectContainer, 'g', 'lasso-container'); - var paths = container.selectAll('path[instance="' + instance + '"]'); - - // update - paths = paths.data(allPoints); - - // remove excess - var pExit = paths.exit().remove(); - // add needed paths - var pEnter = paths.enter().append('path'); - - // merge - paths = paths.merge(pEnter).transition().duration(transitionDuration).ease(easeFunc); - applyPathAttributes(paths); - } - - function applyPathAttributes(path) { - path.attr("class", hypenate(namespace, "lasso-path")).style('opacity', opacity).attr('fill', color).attr("d", line).attr('instance', instance).style("stroke-dasharray", dashArray).attr("stroke", stroke).attr("stroke-width", strokeWidth).style('animation', 'lassoDash ' + animationRate + ' linear').style("animation-iteration-count", "infinite"); - } - - function drag(event) { - /* - effectively create a mouse down and move event (which normally is inteperated - as 'drag' by the browser) by dynamically adding / removing this event on - mouse down / mouse up. - */ - - if (eventCatcher != undefined) { - eventCatcher.dispatch(hypenate(namespace, "drag")); - } - // d3.dispatch(hypenate(namespace,"drag")) - - if (event.which != 1) { - return; - } // ensures left mouse button set - d3.event = event; - var pt = d3.mouse(objectContainer.node()); - var pt = d3.mouse(svg.node()); - - if (xScale != undefined) { - pt[0] = xScale.invert(pt[0]); - } - if (yScale != undefined) { - pt[1] = yScale.invert(pt[1]); - } - pt[0] = pt[0] - chartOffset[0] - objectsOffset[0]; - pt[1] = pt[1] - chartOffset[1] - objectsOffset[1]; - - /* if we have a point already, test if it passes a minimum distance to prevent overwhelming with too many tick functions */ - if (currentPoints.length) { - var lastPt = currentPoints[currentPoints.length - 1]; - var a = [pt[0], pt[1]], - b = [lastPt[0], lastPt[1]]; - - if (xScale) { - b[0] = xScale(b[0]);a[0] = xScale(a[0]); - } - if (yScale) { - b[1] = yScale(b[1]);a[1] = yScale(a[1]); - } - - var dist = euclideanDistance(b, a); - if (dist > tickDistance) { - tick(pt); - } - } else { - tick(pt); - } - } - - function tick(pt) { - /* - If a point is provided update data and objects. - Otherwise just call on data we already have. - Why like this?: - 1. currentPoints is current points to allow disjunct lassos, currentPoints is only pushed to - allPoints after mouseup. - 2. to allow render of objects in the lasso class / updating the data list - just by toggling the button - */ - - if (pt != undefined) { - currentPoints.push(pt); - path.attr("d", line); - if (currentPoints.length < 3) { - return; - } // need at least 3 points to detect anything. - detect(allPoints.concat([currentPoints])); - } else { - detect(allPoints); - } - } - - function detect(lassos) { - if (lassos == undefined) { - lassos = allPoints; - } - objectContainer.selectAll(objectClass).each(function (d, i) { - var current = d3.select(this), - box = current.absolutePosition(), - - // box = current.relativePositionTo(objectContainer.node()), - - boxPts = [[box.left - chartOffset[0] - objectsOffset[0], box.top - chartOffset[1] - objectsOffset[1]], [box.right - chartOffset[0] - objectsOffset[0], box.top - chartOffset[1] - objectsOffset[1]], [box.left - chartOffset[0] - objectsOffset[0], box.bottom - chartOffset[1] - objectsOffset[1]], [box.right - chartOffset[0] - objectsOffset[0], box.bottom - chartOffset[1] - objectsOffset[1]]]; - - if (xScale != undefined) { - boxPts[0][0] = xScale.invert(boxPts[0][0]); - boxPts[1][0] = xScale.invert(boxPts[1][0]); - boxPts[2][0] = xScale.invert(boxPts[2][0]); - boxPts[3][0] = xScale.invert(boxPts[3][0]); - } - if (yScale != undefined) { - boxPts[0][1] = yScale.invert(boxPts[0][1]); - boxPts[1][1] = yScale.invert(boxPts[1][1]); - boxPts[2][1] = yScale.invert(boxPts[2][1]); - boxPts[3][1] = yScale.invert(boxPts[3][1]); - } - - /* - flag needed as we have to test multiple lasso segments, and if the point - is not in one segment, it does not mean it is not in any - */ - var inAnyLassoQ = false; - for (var i = 0; i < lassos.length; i++) { - var lassoPoints = lassos[i]; - // .map(function(pt){ - // var x, y = pt - // if (xScale!=undefined) {x = xScale(x)} - // if (yScale!=undefined) {y = yScale(y)} - // return [x, y] - // }) - var boxInLassoQ = boxPts.every(function (coord) { - return d3.polygonContains(lassoPoints, coord); - }); - - if (boxInLassoQ) { - inAnyLassoQ = true; - } // only update flag in the positive case. - } - - current.classed('in-lasso', inAnyLassoQ); - current.classed('in-lasso-' + instance, inAnyLassoQ); - }); - - updateObjects(); - return objectContainer.selectAll('.in-lasso-' + instance); - } - - function updateObjects() { - objectContainer.selectAll(objectClass).each(function (d, i) { - var t = d3.select(this); - applyObjectAttributes(t, t.classed('in-lasso')); - }); - } - - function applyObjectAttributes(obj, setQ) { - var preLassoFill = obj.attr('_pre_lasso_fill'), - preLassoStroke = obj.attr('_pre_lasso_stroke'), - preLassoStrokeWidth = obj.attr('_pre_lasso_stroke-width'); - - if (setQ) { - obj.classed("in-lasso", true); - obj.classed('in-lasso-' + instance, true); - if (preLassoFill == undefined) { - obj.attr('_pre_lasso_fill', obj.attr('fill')); - } - if (preLassoStroke == undefined) { - obj.attr('_pre_lasso_stroke', obj.attr('stroke')); - } - if (preLassoStrokeWidth == undefined) { - obj.attr('_pre_lasso_stroke-width', obj.attr('stroke-width')); - } - - obj - //BUG: when .raise() - .attr('fill', lassoedFill).attr('stroke', lassoedStroke).attr('stoke-width', lassoedStrokeWidth); - } else { - obj.classed("in-lasso", false); - obj.classed('in-lasso-' + instance, false); - if (preLassoFill != undefined) { - obj.attr('fill', preLassoFill); - } - if (preLassoStroke != undefined) { - obj.attr('stroke', preLassoStroke); - } - if (preLassoStrokeWidth != undefined) { - obj.attr('stroke-width', preLassoStrokeWidth); - } - } - } - - function keyFrames() { - var style = d3.select("html").select('style.' + hypenate(namespace, "lasso-dash")); - if (style.empty()) { - d3.select("html").append('style').classed(hypenate(namespace, "lasso-dash"), true).html("@keyframes lassoDash {to { stroke-dashoffset: 1000;}}"); - } - } - return lasso; - } - - function lassoWidget(selection) { - var namespace = 'd3sm-lasso', - selection = selection, - svg, - chartContainer, - objectContainer, - objectClass, - lassoLine = d3.line().x(function (d, i) { - return d[0]; - }).y(function (d, i) { - return d[1]; - }).curve(d3.curveLinearClosed), - colorFunction$$1 = colorFunction(), - maxNumberOfGroups, - dataExtractor, - xScale, - yScale; - - function onError(msg, style) { - console.log(msg, style); - } - // setup the container - setupDataselectContainer(); - - lassoWidget.objectClass = function (_) { - return arguments.length ? (objectClass = '.' + _.replace('.', ''), lassoWidget) : objectClass; - }; - lassoWidget.svg = function (_) { - return arguments.length ? (svg = _, lassoWidget) : svg; - }; - lassoWidget.submit = function (_) { - return arguments.length ? (submit = _, lassoWidget) : submit; - }; - lassoWidget.maxNumberOfGroups = function (_) { - return arguments.length ? (maxNumberOfGroups = _, lassoWidget) : maxNumberOfGroups; - }; - lassoWidget.onError = function (_) { - return arguments.length ? (onError = _, lassoWidget) : onError; - }; - lassoWidget.objectContainer = function (_) { - return arguments.length ? (objectContainer = _, lassoWidget) : objectContainer; - }; - lassoWidget.chartContainer = function (_) { - return arguments.length ? (chartContainer = _, lassoWidget) : chartContainer; - }; - lassoWidget.dataExtractor = function (_) { - return arguments.length ? (dataExtractor = _, lassoWidget) : dataExtractor; - }; - lassoWidget.xScale = function (_) { - return arguments.length ? (xScale = _, lassoWidget) : xScale; - }; - lassoWidget.yScale = function (_) { - return arguments.length ? (yScale = _, lassoWidget) : yScale; - }; - - function styles() { - var s = d3.select("html").select("style." + namespace + 'lasso-widget'); - if (s.empty()) { - d3.select("html").append("style").classed(namespace + 'lasso-widget', true).html("." + hypenate(namespace, "data-table") + "{\ - height:100px;\ - overflow:auto;\ - }"); - } - } - - function lassoWidget() { - styles(); - } - - function submit(data) { - console.log(data); - } - - function plusTab(tabList) { - var tab = safeSelect(tabList, 'li', hypenate(namespace, 'plus-tab')).classed('ml-auto', true).classed('nav-item', true).on('click', makeNewGroup), - anchor = safeSelect(tab, 'a', 'nav-link'), - icon = safeSelect(anchor, 'i', 'fa').classed('fa-plus fa-2x text-success', true); - } - - function sendTab(tabList) { - var tab = safeSelect(tabList, 'li', hypenate(namespace, 'send-tab')).classed('ml-auto', true).classed('nav-item', true).on('click', clickSend), - anchor = safeSelect(tab, 'a', 'nav-link'), - icon = safeSelect(anchor, 'i', 'fa').classed('fa-paper-plane-o fa-2x text-primary', true); - } - - function clickSend() { - var data = gatherDataLists(); - submit(data); - } - - function closeTab(tabList) { - var tab = safeSelect(tabList, 'li', hypenate(namespace, 'close-tab')).classed('ml-auto', true).classed('nav-item', true).on('click', function () { - var m = d3.mouse(d3.select("html").node()); - }), - anchor = safeSelect(tab, 'a', 'nav-link'), - icon = safeSelect(anchor, 'i', 'fa').classed('fa-window-close-o fa-2x text-danger', true); - } - - function setupDataselectContainer() { - var card = safeSelect(selection, 'div', 'card'), - cardHeader = safeSelect(card, 'div', 'card-header'), - tabList = safeSelect(cardHeader, 'ul', 'nav').classed('nav-tabs card-header-tabs', true).classed(hypenate(namespace, 'tab-list'), true).attr('role', 'tablist'), - cardBody = safeSelect(card, 'div', 'card-body'), - tabContent = safeSelect(cardBody, 'div', 'tab-content').classed(hypenate(namespace, 'tab-content'), true), - - - // leftTabs = safeSelect(tabList, 'div', 'nav mr-auto left-aligned-tabs') - rightTabs = safeSelect(tabList, 'div', 'right-aligned-tabs').classed('nav ml-auto ', true); - - plusTab(rightTabs); - sendTab(rightTabs); - closeTab(rightTabs); - var defaultTab = safeSelect(tabContent, 'div', 'tab-pane').classed(hypenate(namespace, 'default-tab'), true).classed("active", true).classed('text-left', true), - p = safeSelect(defaultTab, 'div').html("Click to add a new group.
" + "Click to submit for re-analysis.
" + "Click to close the dataselect widget."); - // .text('Click the green plus to add a new group.') - } - - function makeRemoveButton(paneBtnList) { - var btn = safeSelect(paneBtnList, 'button', 'remove-btn').classed('btn btn-danger', true).on('click', removeBtnClickToRemove), - i = safeSelect(btn, 'i', 'fa').classed('fa-trash-o', true), - s = safeSelect(btn, 'span').text('Remove group'); - return btn; - } - - function removeBtnClickToRemove(d, i) { - var tab = selection.select('#' + hypenate(namespace, 'tab', d)), - pane = selection.select('#' + hypenate(namespace, 'tab', 'pane', d)); - - showRemainingPanes(); - updateTabAndPaneNumbers(); - } - - function togglePaneById(id) { - var panes = selection.select('.' + hypenate(namespace, 'tab-content')); - } - - function insertTab(tabs, n) { - - // var li = tabs.insert("li", ':nth-child('+(n)+')') - var li = tabs.insert("li", '.right-aligned-tabs').classed('nav-item', true).classed('alert', true).classed('alert-secondary', true).classed('alert-dismissable', true).classed('fade', true).classed('show', true) - // .classed('mr-auto', true) - .attr("role", 'alert').attr("id", hypenate(namespace, 'tab', n)).classed(hypenate(namespace, 'tab'), true); - - var a = safeSelect(li, 'a').attr('data-toggle', 'tab').text('Group ' + n).attr('href', '#' + hypenate(namespace, 'tab', 'pane', n)).on('dblclick', function (d, i) { - var t = d3.select(this); - t.attr('contenteditable', true); - d3.select(t.node().parentNode).classed('alert-secondary', false).classed('alert-warning', true); - - // d3.select("html").on("click", dispatchBlur) - // - // function dispatchBlur(d, i) { - // if (d3.event.target != d3.select(this)) { - // d3.select(this).dispatch("blur") - // } - // } - - }).on('blur', function (d, i) { - - var t = d3.select(this); - t.attr('contenteditable', false); - d3.select(t.node().parentNode).classed('alert-secondary', true).classed('alert-warning', false); - }).on('input', function (d, i) { - var t = d3.select(this); - var txt = t.text(); - d3.select(t.attr('href')).select("p.lead").text(txt); - }); - - return li; - } - - function makeTabPane(panes, n) { - var pane = panes.append('div').datum(n).attr('role', 'tabpanel').classed('tab-pane', true).classed('text-left', true).classed(hypenate(namespace, 'tab', 'pane'), true).attr('id', hypenate(namespace, 'tab', 'pane', n)); - return pane; - } - - function populatePane(pane, n) { - var lead = safeSelect(pane, 'p', 'lead').text('Group ' + n), - tableContainer = safeSelect(pane, 'div', 'table-responsive').attr("class", hypenate(namespace, "data-table")), - table = safeSelect(tableContainer, "table", "table").classed("table-sm", true).classed("table-hover", true), - caption = safeSelect(table, "caption", "caption").html("List of selected"), - btns = safeSelect(pane, 'div', 'text-right'), - cN = n % colorFunction$$1.colors().length; - if (n % 2 != 0) { - cN = colorFunction$$1.colors().length - 1 - cN; - } - - var lassoBtn = makeLassoButton(btns); - var currentLasso = makeLassoFunction(n, colorFunction$$1.colors()[cN]); - var clearBtn = makeClearButton(btns); - - bindButtons(lassoBtn, clearBtn, currentLasso, table); - - var removeBtn = makeRemoveButton(btns); - - lead.on("mouseover", function () { - currentLasso.draw(); - }); - lead.on("mouseout", function () { - currentLasso.remove(); - }); - } - - function makeLassoButton(btns) { - var btn = safeSelect(btns, 'button', 'lasso-btn').classed('btn btn-info', true).classed(namespace, true); - // .datum([lasso]) - var i = safeSelect(btn, 'i', 'fa').classed('fa-hand-pointer-o', true); - var s = safeSelect(btn, 'span').text('Lasso select'); - return btn; - } - - function makeClearButton(btns) { - var btn = safeSelect(btns, 'button', 'clear-btn').classed('btn btn-light', true).classed(namespace, true); - var i = safeSelect(btn, 'i', 'fa').classed('fa-eraser', true); - var s = safeSelect(btn, 'span').text('Clear selection'); - return btn; - } - - function makeLassoFunction(instance, color) { - var currentLasso = lasso().namespace(namespace).svg(svg).objectClass(objectClass).chartContainer(chartContainer).objectContainer(objectContainer).instance(instance).color(color).yScale(yScale).xScale(xScale); - - return currentLasso; - } - - function bindButtons(lassoBtn, clearBtn, currentLasso, table) { - currentLasso.eventCatcher(lassoBtn); - lassoBtn.node().addEventListener("click", lassoBtnToggle); - - lassoBtn.datum([currentLasso]).on(hypenate(namespace, 'render'), function () { - d3.select(this).datum()[0].draw(); - }).on(hypenate(namespace, "drag"), function (las) { - var nodes = chartContainer.selectAll(".in-lasso-" + las[0].instance()).nodes(); - var tableData = nodes.map(function (d, i) { - var extracted = dataExtractor(d3.select(d).datum()); - extracted["__node"] = d; - return extracted; - }); - - makeTable(table, tableData, currentLasso); - }); - - clearBtn.on('click', function () { - currentLasso.allPoints([]); - currentLasso.currentPoints([]); - lassoBtn.dispatch(hypenate(namespace, "drag")); - }); - } - - function lassoBtnToggle() { - var that = d3.select(this); - var las = that.datum()[0]; - - las.toggle(); - var activeQ = las.activeQ(); - - if (las.activeQ()) { - that.classed('btn-info', !activeQ); - that.classed('btn-warning', activeQ); - that.select("span").text("Lasso select (active)"); - selection.selectAll("." + namespace + ".lasso-btn").dispatch(hypenate(namespace, 'render')); - d3.select("html").node().addEventListener('mousedown', monitorLassoButtonState); - } else { - that.classed('btn-info', !activeQ); - that.classed('btn-warning', activeQ); - that.select("span").text("Lasso select"); - d3.select("html").node().removeEventListener('mousedown', monitorLassoButtonState); - } - - function monitorLassoButtonState(event) { - /* - activeLasso stops event stopPropagation, so this event will not register in - that case. Thus we only need to ensure that the usre is clicking on the lasso button - otherwise, de-spawn lasso. - */ - if (event.target != that.node() && event.target != that.select("span").node() && event.target != that.select("i").node()) { - // event.preventDefault() - // event.stopPropagation() - las.toggle(false); - that.classed('btn-info', true); - that.classed('btn-warning', false); - that.select("span").text("Lasso select"); - // console.log(that, that.node()) - // that.dispatch("focusout") - } - } - } - - function updateTableHeaderColumns(headR, tableData) { - var headerKeys = d3.keys(tableData[0]).filter(function (k) { - return k != "__node"; - }); - if (headerKeys.length > 0) { - // headerKeys = ["remove"].concat(headerKeys) - headerKeys.push("remove"); - } - - var headerCols = headR.selectAll("th"); - - headerCols = headerCols.data(headerKeys); - headerCols.exit().remove(); - headerCols = headerCols.merge(headerCols.enter().append("th").attr("scope", "col")).text(function (d, i) { - return d; - }); - - return headerKeys; - } - - function updateTableRows(body, tableData) { - var bodyRows = body.selectAll("tr"); - - bodyRows = bodyRows.data(tableData); - bodyRows.exit().remove(); - bodyRows = bodyRows.merge(bodyRows.enter().append("tr")); - - return bodyRows; - } - - function updateTableRowColumns(cols, headerKeys, rowData, tableData, lasso$$1, table) { - cols = cols.data(headerKeys); - cols.exit().remove(); - cols = cols.merge(cols.enter().append("td")); - - cols.html(function (d, i) { - if (d != "remove") { - return rowData[d]; - } - return ""; - }).classed("text-left", function (d, i) { - if (d != "remove") { - return false; - } - return true; - }); - // .attr("scope",function(d, i){ - // if (d != "remove") { return false } - // return true - // }) - - cols.select("i.fa-close").on("click", function (d, i) { - var node = rowData["__node"], - n = d3.select(node); - n.classed("in-lasso", false); - n.classed("in-lasso-" + lasso$$1.instance(), false); - - tableData.map(function (dd, j) { - if (dd["__node"] == node) { - tableData.splice(j, 1); - makeTable(table, tableData, lasso$$1); - } - }); - }); - } - - function makeTable(table, tableData, lasso$$1) { - var head = safeSelect(table, "thead"), - headR = safeSelect(head, "tr"), - body = safeSelect(table, "tbody"), - headerKeys = updateTableHeaderColumns(headR, tableData), - bodyRows = updateTableRows(body, tableData); - - bodyRows.each(function (rowData, i) { - var t = d3.select(this); - var cols = t.selectAll("td"); - - updateTableRowColumns(cols, headerKeys, rowData, tableData, lasso$$1, table); - - t.on("mouseover", function (d, i) { - lasso$$1.applyObjectAttributes(d3.select(d["__node"]), true); - }).on("mouseout", function (d, i) { - lasso$$1.applyObjectAttributes(d3.select(d["__node"]), false); - }); - }); - } - - function makeNewGroup(d, i) { - - var tabs = selection.select('.' + hypenate(namespace, 'tab-list')), - panes = selection.select('.' + hypenate(namespace, 'tab-content')), - n = newGroupNumber(); - - if (tabs.selectAll('.' + hypenate(namespace, 'tab')).size() == maxNumberOfGroups) { - onError('only ' + maxNumberOfGroups + ' allowed.', 'warning'); - return; - } - - var tab = insertTab(tabs, n), - pane = makeTabPane(panes, n); - - populatePane(pane, n); - togglePaneById(pane.attr('id')); - } - - function newGroupNumber() { - var tabs = selection.select('.' + hypenate(namespace, 'tab-list')); //, - var n = tabs.selectAll('li').size() - 3, - // minus 1 for plus tab - s = 'Group ' + n, - gs = []; - while (gs.includes(s)) { - n += 1;s = 'Group ' + n; - } - return n; - } - - function updateTabAndPaneNumbers() { - var tabs = selection.select('.' + hypenate(namespace, 'tab-list')), - panes = selection.select('.' + hypenate(namespace, 'tab-content')); - - tabs = tabs.selectAll('.' + hypenate(namespace, 'tab')); - panes = panes.selectAll('.' + hypenate(namespace, 'tab', 'pane')); - - tabs.each(function (d, i) { - d3.select(this).datum(i).attr("id", hypenate(namespace, 'tab', i)).select('a').attr('href', '#' + hypenate(namespace, 'tab', 'pane', i)).text(function (dd, ii) { - var curText = d3.select(this).text(); - if (curText.split(' ')[0] == 'Group') { - return 'Group ' + i; - } - return curText; - }); - }); - - panes.each(function (d, i) { - d3.select(this).datum(i).attr('id', hypenate(namespace, 'tab', 'pane', i)); - safeSelect(d3.select(this), 'p', 'lead').text(function (dd, ii) { - var curText = d3.select(this).text(); - if (curText.split(' ')[0] == 'Group') { - return 'Group ' + i; - } - return curText; - }); - safeSelect(d3.select(this), 'button', 'remove-btn').on('click', removeBtnClickToRemove); - }); - } - - function gatherDataLists() { - var tabs = selection.select('.' + hypenate(namespace, 'tab-list')); - var panes = selection.select('.' + hypenate(namespace, 'tab-content')); - var tables = panes.selectAll('.' + hypenate(namespace, "data-table")); - var data = {}; - - return data; - } - - function showRemainingPanes() { - var tabs = selection.select('.' + hypenate(namespace, 'tab-list')), - panes = selection.select('.' + hypenate(namespace, 'tab-content')), - remainingTabs = tabs.selectAll('.' + hypenate(namespace, 'tab')); - - if (remainingTabs.size() == 0) ; else { - var lastTab = remainingTabs.nodes()[remainingTabs.size() - 1], - lastPaneId = d3.select(lastTab).select('a').attr('href'); - } - } - - return lassoWidget; - } - - function upset(selection) { - var data, - orient = 'horizontal', - spaceX, - spaceY, - overflowQ = false, - minObjectSize = 20, - maxObjectSize = 50, - circleStrokeWidth = 2, - - // colorFunction= - backgroundFill = 'transparent', - namespace = 'd3sm-upset', - objectClass = 'upset', - transitionDuration = 1000, - easeFunc = d3.easeExp, - setKey = "set", - intersectionKey = "intersection", - elementsKey = "elements", - setExtractor = function setExtractor(key, i) { - return data[key][setKey]; - }, - intersectionExtractor = function intersectionExtractor(key, i) { - return data[key][intersectionKey]; - }, - elementExtractor = function elementExtractor(key, i) { - return data[key][elementsKey]; - }, - cellKeys, - setValues, - intersectionValues, - xObjectSpacer = 0.05, - yObjectSpacer = 0.05, - radius, - - - // listDelim = ';' - - yObjectSize, - ySpacerSize, - xObjectSize, - xSpacerSize, - setKeySortingFunction = function setKeySortingFunction(a, b) { - return setValues.indexOf(setExtractor(a)) - setValues.indexOf(setExtractor(b)); - }, - intersectionKeySortingFunction = function intersectionKeySortingFunction(a, b) { - return intersectionValues.indexOf(intersectionExtractor(a)) - intersectionValues.indexOf(intersectionExtractor(b)); - }; - - upset.selection = function (_) { - return arguments.length ? (selection = _, upset) : selection; - }; - upset.data = function (_) { - return arguments.length ? (data = _, upset) : data; - }; - upset.orient = function (_) { - return arguments.length ? (orient = _, upset) : orient; - }; - upset.spaceX = function (_) { - return arguments.length ? (spaceX = _, upset) : spaceX; - }; - upset.spaceY = function (_) { - return arguments.length ? (spaceY = _, upset) : spaceY; - }; - upset.overflowQ = function (_) { - return arguments.length ? (overflowQ = _, upset) : overflowQ; - }; - upset.minObjectSize = function (_) { - return arguments.length ? (minObjectSize = _, upset) : minObjectSize; - }; - upset.maxObjectSize = function (_) { - return arguments.length ? (maxObjectSize = _, upset) : maxObjectSize; - }; - upset.circleStrokeWidth = function (_) { - return arguments.length ? (circleStrokeWidth = _, upset) : circleStrokeWidth; - }; - upset.backgroundFill = function (_) { - return arguments.length ? (backgroundFill = _, upset) : backgroundFill; - }; - upset.namespace = function (_) { - return arguments.length ? (namespace = _, upset) : namespace; - }; - upset.objectClass = function (_) { - return arguments.length ? (objectClass = _, upset) : objectClass; - }; - upset.transitionDuration = function (_) { - return arguments.length ? (transitionDuration = _, upset) : transitionDuration; - }; - upset.easeFunc = function (_) { - return arguments.length ? (easeFunc = _, upset) : easeFunc; - }; - upset.cellKeys = function (_) { - return arguments.length ? (cellKeys = _, upset) : cellKeys; - }; - upset.setValues = function (_) { - return arguments.length ? (setValues = _, upset) : setValues; - }; - upset.intersectionValues = function (_) { - return arguments.length ? (intersectionValues = _, upset) : intersectionValues; - }; - upset.xObjectSpacer = function (_) { - return arguments.length ? (xObjectSpacer = _, upset) : xObjectSpacer; - }; - upset.yObjectSpacer = function (_) { - return arguments.length ? (yObjectSpacer = _, upset) : yObjectSpacer; - }; - upset.radius = function (_) { - return arguments.length ? (radius = _, upset) : radius; - }; - upset.setExtractor = function (_) { - return arguments.length ? (setExtractor = _, upset) : setExtractor; - }; - upset.intersectionExtractor = function (_) { - return arguments.length ? (intersectionExtractor = _, upset) : intersectionExtractor; - }; - upset.elementExtractor = function (_) { - return arguments.length ? (elementExtractor = _, upset) : elementExtractor; - }; - upset.setKeySortingFunction = function (_) { - return arguments.length ? (setKeySortingFunction = _, upset) : setKeySortingFunction; - }; - upset.intersectionKeySortingFunction = function (_) { - return arguments.length ? (intersectionKeySortingFunction = _, upset) : intersectionKeySortingFunction; - }; - - upset.yObjectSize = function (_) { - return arguments.length ? (yObjectSize = _, upset) : yObjectSize; - }; - upset.ySpacerSize = function (_) { - return arguments.length ? (ySpacerSize = _, upset) : ySpacerSize; - }; - upset.xObjectSize = function (_) { - return arguments.length ? (xObjectSize = _, upset) : xObjectSize; - }; - upset.xSpacerSize = function (_) { - return arguments.length ? (xSpacerSize = _, upset) : xSpacerSize; - }; - - function upset() { - // for convenience in handling orientation specific values - var horizontalQ = orient == 'horizontal' ? true : false; - var verticalQ = !horizontalQ; - - // background cliping rectangle - var bgcpRect = { x: 0, y: 0, width: spaceX, height: spaceY }; - var container = setupContainer(selection, namespace, bgcpRect, backgroundFill); - - cellKeys = d3.keys(data); - setValues = unique(cellKeys.map(setExtractor)).sort(); - intersectionValues = unique(cellKeys.map(intersectionExtractor)).sort().sort(function (a, b) { - return a.split(';').length - b.split(';').length; - }); - - if (!horizontalQ) { - cellKeys.sort(function (a, b) { - return setKeySortingFunction(a, b) || intersectionKeySortingFunction(a, b); - }); - } else { - cellKeys.sort(function (a, b) { - return intersectionKeySortingFunction(a, b) || setKeySortingFunction(a, b); - }); - } - - var xValues = horizontalQ ? intersectionValues : setValues, - yValues = horizontalQ ? setValues : intersectionValues, - xDim = horizontalQ ? xValues.length : yValues.length, - yDim = horizontalQ ? yValues.length : xValues.length; - - // console.log(xValues, yValues) - - - xObjectSize = calculateWidthOfObject(spaceX, xDim, minObjectSize, maxObjectSize, xObjectSpacer, overflowQ); - yObjectSize = calculateWidthOfObject(spaceY, yDim, minObjectSize, maxObjectSize, yObjectSpacer, overflowQ); - xSpacerSize = calculateWidthOfSpacer(xValues, spaceX, xObjectSize, xDim, xObjectSpacer, overflowQ); - ySpacerSize = calculateWidthOfSpacer(yValues, spaceY, yObjectSize, yDim, yObjectSpacer, overflowQ); - - var ySpacer = groupingSpacer().horizontalQ(false).moveby('category').numberOfObjects(yDim).objectSize(yObjectSize).spacerSize(ySpacerSize).transitionDuration(transitionDuration).easeFunc(easeFunc); - - var xSpacer = groupingSpacer().horizontalQ(true).moveby('category').numberOfObjects(xDim).objectClass(objectClass).objectSize(xObjectSize).spacerSize(xSpacerSize).transitionDuration(transitionDuration).easeFunc(easeFunc); - - if (verticalQ) { - xSpacer.objectClass(objectClass); - ySpacer.namespace('across').objectClass(hypenate(objectClass, 'across')); - - ySpacer(container, yValues, 0); - container.selectAll('g.' + hypenate(objectClass, 'across')).each(function (d, i) { - xSpacer(d3.select(this), xValues, 0); - }); - } else { - xSpacer.namespace('across').objectClass(hypenate(objectClass, 'across')); - ySpacer.objectClass(objectClass); - - xSpacer(container, xValues, 0); - container.selectAll('g.' + hypenate(objectClass, 'across')).each(function (d, i) { - ySpacer(d3.select(this), yValues, 0); - }); - } - - var cells = container.selectAll('g:not(.to-remove).' + objectClass); - var lookup = {}; - cellKeys.map(function (k, i) { - lookup[setExtractor(k) + '::' + intersectionExtractor(k)] = k; - }); - - // var positionedCellKeys = [] - // for (var i = 0; i < setValues.length; i++) { - // for (var j = 0; j < intersectionValues.length; j++) { - // var lookupValue = lookup[setValues[j]+"::"+intersectionValues[i]] - // if (lookupValue == undefined) { - // positionedCellKeys.push(undefined) - // } else { - // positionedCellKeys.push(lookupValue) - // } - // console.log(i, j, lookupValue) - // } - // } - // - // // console.log(positionedCellKeys) - // - // cells.data(positionedCellKeys); - - - cells.data(cellKeys); - - cells.each(function (key, i) { - var t = d3.select(this); - if (key == undefined) { - return; - } - var currentData = data[key]; - // console.log(key, currentData) - var set = setExtractor(key, i), - intersection = intersectionExtractor(key, i); - - // console.log(set, intersection) - - t.classed(intersection, true); - t.classed(set, true); - - var c = safeSelect(t, 'circle', hypenate(objectClass, 'circle')); - c.attr('cx', xObjectSize / 2).attr('cy', yObjectSize / 2).attr('r', radius == undefined ? Math.min(xObjectSize, yObjectSize) / 2 : radius).attr('fill', intersection.includes(set) ? "black" : 'rgb(233,233,233)').attr('stroke', "black").attr("in-intersection", intersection.includes(set)); - }); - } - - function intersectionTotals() { - var totals = {}; - // intersectionValues.sort(function(a, b){ return intersectionKeySortingFunction(a, b) }) - - intersectionValues.map(function (k, i) { - totals[k] = { 'total': 0 }; - }); - cellKeys.map(function (k, i) { - var e = elementExtractor(k, i); - if (totals[intersectionExtractor(k, i)]['total'] == 0) { - if (Array.isArray(e)) { - totals[intersectionExtractor(k, i)]['total'] += e.length; - totals[intersectionExtractor(k, i)]['values'] = e; - } else { - totals[intersectionExtractor(k, i)]['total'] += e; - } - } - }); - return totals; - } - - function setTotals() { - var totals = {}; - // intersectionValues.sort(function(a, b){ return intersectionKeySortingFunction(a, b) }) - - setValues.map(function (k, i) { - totals[k] = { 'total': 0 }; - }); - - cellKeys.map(function (k, i) { - var e = elementExtractor(k, i); - if (Array.isArray(e)) { - totals[setExtractor(k, i)]['total'] += e.length; - } else { - totals[setExtractor(k, i)]['total'] += e; - } - }); - return totals; - } - - upset.intersectionTotals = intersectionTotals; - upset.setTotals = setTotals; - - return upset; - } - - function filterTable(selection) { - var data, - namespace = 'd3sm-filter-table', - fieldFunction = function fieldFunction(record, columnKey, recordValue) { - return recordValue; - }; - - filterTable.data = function (_) { - return arguments.length ? (data = _, filterTable) : data; - }; - filterTable.namespace = function (_) { - return arguments.length ? (namespace = _, filterTable) : namespace; - }; - filterTable.fieldFunction = function (_) { - return arguments.length ? (fieldFunction = _, filterTable) : fieldFunction; - }; - - function filterTable() { - - var container = safeSelect(selection, 'div', 'filter-table').classed(hypenate(namespace, 'container'), true), - inputGroup = safeSelect(container, 'div', 'filter-input-group').classed('input-group', true), - inputPrepend = safeSelect(inputGroup, 'div', 'input-group-prepend'), - inputPrependSpan = safeSelect(inputPrepend, 'span', 'input-group-text').classed('search-button', true), - inputPrependSpanIcon = safeSelect(inputPrependSpan, 'i', 'fa fa-search'), - input = safeSelect(inputGroup, 'input', 'form-control').attr('placeholder', 'filter').attr('type', 'text'), - inputAppend = safeSelect(inputGroup, 'div', 'input-group-append'), - inputAppendButton = safeSelect(inputAppend, 'a', 'close-button').classed('btn btn-outline-secondary', true), - inputAppendButtonIcon = safeSelect(inputAppendButton, 'i', 'fa fa-close'), - closeButton = inputAppendButton, - rTable = safeSelect(container, 'div', 'table-responsive'), - table = safeSelect(rTable, 'table', 'table').classed('table-hover', true).classed('table-bordered', true).classed("table-striped", true), - tHead = safeSelect(table, 'thead').classed("thead-dark", true), - header = safeSelect(tHead, 'tr'), - tBody = safeSelect(table, 'tbody'); - - var rows = d3.keys(data), - cols = d3.keys(data[rows[0]]); - - var th = makeColumns(header, cols); - var tr = makeRows(tBody, rows); - var tr = populateRecords(tr, cols); - - input.on('input', function (d, i) { - var val = input.property('value'), - reg = new RegExp(val, 'gi'), - use; - if (val == '') { - use = rows; - } else { - use = []; - rows.map(function (r, i) { - var row = data[r]; - var fieldJoin = cols.map(function (c, j) { - return fieldFunction(row, c, row[c]); - // return row[c] - }).join(''); - var match = fieldJoin.match(reg); - if (match == null || match.join('') == '') ; else { - use.push(r); - } - }); - } - - tr = makeRows(tBody, use); - tr = populateRecords(tr, cols); - }); - - closeButton.on('click', function (d, i) { - input.property('value', '').dispatch('input'); - }); - } - - function makeColumns(header, cols) { - var th = header.selectAll('th'); - th = th.data(cols); - th.exit().remove(); - th = th.merge(th.enter().append('th')); - th.attr('scope', 'col').text(function (d, i) { - return d; - }); - return th; - } - - function makeRows(body, rows) { - var tr = body.selectAll('tr'); - tr = tr.data(rows); - tr.exit().remove(); - tr = tr.merge(tr.enter().append('tr')); - return tr; - } - - function populateRecords(rows, cols) { - rows.each(function (r, i) { - var record = data[r], - t = d3.select(this); - - var fields = t.selectAll('td'); - fields = fields.data(cols); - fields.exit().remove(); - fields = fields.merge(fields.enter().append('td')); - - fields.attr('scope', function (f, j) { - return j == 0; - }).html(function (f, j) { - return fieldFunction(record, f, record[f]); - }); - }); - return rows; - } - - return filterTable; - } - - // Import styles (automatically inject into ). - - // /** @module d3sm */ - var d3sm$1 = {}; - d3sm$1.axis = axis; - d3sm$1.bar = bar; - d3sm$1.bubbleHeatmap = bubbleHeatmap; - d3sm$1.heatmap = heatmap; - d3sm$1.boxwhisker = boxwhisker; - d3sm$1.colorFunction = colorFunction; - d3sm$1.datatoggle = datatoggle; - d3sm$1.groupingSpacer = groupingSpacer; - d3sm$1.tooltip = tooltip; - d3sm$1.scatter = scatter; - d3sm$1.plotZoom = plotZoom; - d3sm$1.multiPlotZoom = multiPlotZoom; - d3sm$1.violin = violin; - d3sm$1.numericLegend = numericLegend; - d3sm$1.categoricLegend = categoricLegend; - d3sm$1.lasso = lasso; - d3sm$1.lassoWidget = lassoWidget; - d3sm$1.selectFilter = selectFilter; - d3sm$1.upset = upset; - d3sm$1.filterTable = filterTable; - - d3sm$1.uniqueElements = uniqueElements; - d3sm$1.getTranslation = getTranslation; - d3sm$1.modifyHexidecimalColorLuminance = modifyHexidecimalColorLuminance; - d3sm$1.tickRange = tickRange; - d3sm$1.quartiles = quartiles; - d3sm$1.extractViolinValues = extractViolinValues; - d3sm$1.hypenate = hypenate; - d3sm$1.round = round; - d3sm$1.getContainingSVG = getContainingSVG; - d3sm$1.interpolateColors = interpolateColors; - d3sm$1.truncateText = truncateText; - d3sm$1.safeSelect = safeSelect; - - d3sm$1.whichBin = whichBin; - d3sm$1.unique = unique; - d3sm$1.flatten = flatten; - - d3sm$1.setupStandardChartContainers = setupStandardChartContainers; - d3sm$1.log = log; - d3sm$1.warn = warn; - d3sm$1.info = info; - d3sm$1.error = error; - d3sm$1.consoleGroup = consoleGroup; - d3sm$1.consoleGroupEnd = consoleGroupEnd; - d3sm$1.resizeDebounce = resizeDebounce; - - d3sm$1.debugQ = false; - - // Import a logger for easier debugging - // import debug from 'debug'; - // const log = debug('app:log'); - - // The logger should only be disabled if we're not in production. - // if ("development" !== 'production') { - // // Enable the logger. - // debug.enable('*'); - // log('Logging is enabled!'); - // - // // Enable LiveReload - // document.write( - // ' - - - - - - - - - - - - - - - - - - - - -
- - - -
-

By Value

-
- -
-

By Category

-
- -
-

By Value Guidelines

-
-
- - - - - diff --git a/demos/axes/js/v001.js b/demos/axes/js/v001.js deleted file mode 100644 index ab67ca4..0000000 --- a/demos/axes/js/v001.js +++ /dev/null @@ -1,196 +0,0 @@ -var data = d3smDemoData['basicViolins'] - -var -selection = d3.select("#axes-by-value") -namespace = 'axes', -container = d3sm.safeSelect(selection, 'svg', namespace).style('width', '1000px'), -lAxis = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'left-axis')).attr('transform', "translate(100,100)"), -rAxis = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'right-axis')).attr('transform', "translate(900,100)") -uAxis = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'up-axis')).attr('transform', "translate(100,100)") -dAxis = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'down-axis')).attr('transform', "translate(100,400)") - - -tickValues = [1,2,3,4,5] - -var uA = d3sm.axis(uAxis) -.spaceX(800) -.spaceY(100) -.orient("top") -.overflowQ(false) -.tickValues(tickValues) -.numberOfTicks(5) -.namespace('u') -// .guideLinesQ(true) -.guidelineSpace(300) - -var dA = d3sm.axis(dAxis) -.spaceX(800) -.spaceY(100) -.orient("bottom") -.overflowQ(false) -.tickValues(tickValues) -.numberOfTicks(5) -.namespace('d') -// .guideLinesQ(true) -.guidelineSpace(300) - - -var lA = d3sm.axis(lAxis) -.spaceX(100) -.spaceY(300) -.orient("left") -.overflowQ(false) -.tickValues(tickValues) -.numberOfTicks(5) -.namespace('l') -// .guideLinesQ(true) -.guidelineSpace(800) - -var rA = d3sm.axis(rAxis) -.spaceX(100) -.spaceY(300) -.orient("right") -.overflowQ(false) -.tickValues(tickValues) -.numberOfTicks(5) -.namespace('r') -// .guideLinesQ(true) -.guidelineSpace(800) - -uA() -dA() -lA() -rA() - - - -var -selection = d3.select("#axes-by-cat") -namespace = 'axes', -container = d3sm.safeSelect(selection, 'svg', namespace).style('width', '1000px'), -lAxis = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'left-axis')).attr('transform', "translate(100,100)"), -rAxis = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'right-axis')).attr('transform', "translate(900,100)") -uAxis = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'up-axis')).attr('transform', "translate(100,100)") -dAxis = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'down-axis')).attr('transform', "translate(100,400)") - - -tickValues = [1,2,3,4,5] -tickLabels = ["this", "this-text", "this-text-gets", "this-text-gets-longer", "this-text-gets-longer-as-we-go-on"] - -var uA = d3sm.axis(uAxis) -.spaceX(800) -.spaceY(100) -.orient("top") -.overflowQ(false) -.categoricalQ(true) -.tickLabels(tickLabels) -.numberOfTicks(5) -.namespace('u-c') -// .guideLinesQ(true) -.guidelineSpace(300) - -var dA = d3sm.axis(dAxis) -.spaceX(800) -.spaceY(100) -.orient("bottom") -.overflowQ(false) -.categoricalQ(true) -.tickLabels(tickLabels) -.numberOfTicks(5) -.namespace('d-c') -// .guideLinesQ(true) -.guidelineSpace(300) - - -var lA = d3sm.axis(lAxis) -.spaceX(100) -.spaceY(300) -.orient("left") -.overflowQ(false) -.categoricalQ(true) -.tickLabels(tickLabels) -.numberOfTicks(5) -.namespace('l-c') -// .guideLinesQ(true) -.guidelineSpace(800) - -var rA = d3sm.axis(rAxis) -.spaceX(100) -.spaceY(300) -.orient("right") -.overflowQ(false) -.categoricalQ(true) -.tickLabels(tickLabels) -.numberOfTicks(5) -.namespace('r-c') -// .guideLinesQ(true) -.guidelineSpace(800) - -uA() -dA() -lA() -rA() - - - -var -selection = d3.select("#axes-by-val-g") -namespace = 'axes', -container = d3sm.safeSelect(selection, 'svg', namespace).style('width', '1000px'), -lAxis = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'left-axis')).attr('transform', "translate(100,100)"), -rAxis = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'right-axis')).attr('transform', "translate(900,100)") -uAxis = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'up-axis')).attr('transform', "translate(100,100)") -dAxis = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'down-axis')).attr('transform', "translate(100,400)") - - -tickValues = [1,2,3,4,5] - -var uA = d3sm.axis(uAxis) -.spaceX(800) -.spaceY(100) -.orient("top") -.overflowQ(false) -.tickValues(tickValues) -.numberOfTicks(5) -.namespace('u-g') -.guideLinesQ(true) -.guidelineSpace(300) - -var dA = d3sm.axis(dAxis) -.spaceX(800) -.spaceY(100) -.orient("bottom") -.overflowQ(false) -.tickValues(tickValues) -.numberOfTicks(5) -.namespace('d-g') -.guideLinesQ(true) -.guidelineSpace(300) - - -var lA = d3sm.axis(lAxis) -.spaceX(100) -.spaceY(300) -.orient("left") -.overflowQ(false) -.tickValues(tickValues) -.numberOfTicks(5) -.namespace('l-g') -.guideLinesQ(true) -.guidelineSpace(800) - -var rA = d3sm.axis(rAxis) -.spaceX(100) -.spaceY(300) -.orient("right") -.overflowQ(false) -.tickValues(tickValues) -.numberOfTicks(5) -.namespace('r-g') -.guideLinesQ(true) -.guidelineSpace(800) - -uA() -dA() -lA() -rA() diff --git a/demos/axes/js/v003.js b/demos/axes/js/v003.js deleted file mode 100644 index e69de29..0000000 diff --git a/demos/bar-chart-same-data-complex-grouping/index.html b/demos/bar-chart-same-data-complex-grouping/index.html deleted file mode 100644 index 84a766e..0000000 --- a/demos/bar-chart-same-data-complex-grouping/index.html +++ /dev/null @@ -1,195 +0,0 @@ - - - - - Complex Grouping - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/demos/bar-chart-same-data-complex-grouping/js/v001.js b/demos/bar-chart-same-data-complex-grouping/js/v001.js deleted file mode 100644 index 50a6405..0000000 --- a/demos/bar-chart-same-data-complex-grouping/js/v001.js +++ /dev/null @@ -1,125 +0,0 @@ -function barPlot( selection ) { - var data, - namespace - plot.data = function(_){return arguments.length ? (data = _, plot) : data; }; - plot.namespace = function(_){return arguments.length ? (namespace = _, plot) : namespace; }; - - - function plot() { - var - databar = d3sm.safeSelect(selection, 'form', d3sm.hypenate(namespace,'databar')), - dataPrint = d3sm.safeSelect(selection, 'div', d3sm.hypenate(namespace,'data-print')), - - selections = d3sm.setupStandardChartContainers(selection, namespace, {w: 1000, h:500}, undefined, {axes:{x:0.1, y:0.1},space:{w:1,h:1}}) - container = selections.svg.selection, - bars = selections.plot.selection, - xAxis = selections.xAxis.selection, - yAxis = selections.yAxis.selection - - - // make data selection - var dataSelect = d3sm.datatoggle(databar) - .keys(d3.keys(data.groupings)) - .updateFunction(function(){plot(); zoom.reset()}) - .namespace(namespace)() - - // grab current selected key - var currentKey = dataSelect.currentKey() - // // extract subdata - // var currentData = data[currentKey] - // - - - dataPrint.text(JSON.stringify(data.groupings[currentKey])) - // make bars - var b = d3sm.bar(bars) - .data(data.data) - .grouping(data.groupings[currentKey]) - .spaceX(selections.plot.rect.w) - .spaceY(selections.plot.rect.h) - .overflowQ(true) - .orient('horizontal') - .maxObjectSize(20) - .minObjectSize(10) - .valueExtractor(function(d, i){return data.data[d]['value'] }) - - - // b.tooltip().keys(['value']) - // - // b.colorFunction().colorBy('index') - b() - - // extract keys and values to safe re-computation - var bk = b.barKeys(), bv = b.barValues() - - // left d3sm.axis - var yA = d3sm.axis(yAxis) - .spaceX(selections.yAxis.rect.w) - .spaceY(selections.yAxis.rect.h) - .orient("left") - .overflowQ(false) - .tickValues(bv) - .guideLinesQ(true) - .guidelineSpace(selections.xAxis.rect.w) - .namespace('axis-left') - .numberOfTicks(5) - yA() - - // bottom d3sm.axis - var xA = d3sm.axis(xAxis) - .spaceX(selections.xAxis.rect.w) - .spaceY(selections.xAxis.rect.h) - .orient("bottom") - .overflowQ(true) - .categoricalQ(true) - .tickLabels(bk) - .namespace('axis-bottom') - .minObjectSize(b.minObjectSize()) - .maxObjectSize(b.maxObjectSize()) - .grouping(data.groupings[currentKey]) - .objectSpacer(b.objectSpacer()) - xA() - - b.selection().selectAll("rect").nodes().map(function(d, i){ - console.log(d.getBoundingClientRect(), i) - return this - }), - console.log( - b.selection().node(), - b.selection().selectAll("rect").nodes(), - b.selection().selectAll("rect").nodes(), - 'BCR:',b.selection().node().getBoundingClientRect(), - 'BBox:',b.selection().node().getBBox() - ) - - - - - - // // set-up zoom, whell and drag - var zoom = d3sm.plotZoom(b, xA, yA).eventType('drag') - .orient('horizontal') - // .xLock( - // objs.getBBox().width - // ) - - console.log(zoom) - - // zoom.setLocks() - - var pan = d3.zoom().scaleExtent([1,1]).on('zoom', function(){ - // zoom.setLocks() - zoom.eventType('drag')() - }) - container.call(pan).on('wheel', function(){ - // zoom.setLocks() - - zoom.eventType('wheel')() - }) - } - - - - - return plot -} diff --git a/demos/basic-violins/index.html b/demos/basic-violins/index.html deleted file mode 100644 index e0cbea7..0000000 --- a/demos/basic-violins/index.html +++ /dev/null @@ -1,172 +0,0 @@ - - - - - Basic Violins - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/demos/box-whiskers-points/index.html b/demos/box-whiskers-points/index.html deleted file mode 100644 index 165e1ac..0000000 --- a/demos/box-whiskers-points/index.html +++ /dev/null @@ -1,154 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/demos/box-whiskers/index.html b/demos/box-whiskers/index.html deleted file mode 100644 index 69abf28..0000000 --- a/demos/box-whiskers/index.html +++ /dev/null @@ -1,155 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/demos/bubble-heatmap/index.html b/demos/bubble-heatmap/index.html deleted file mode 100644 index 19fc41e..0000000 --- a/demos/bubble-heatmap/index.html +++ /dev/null @@ -1,161 +0,0 @@ - - - - - Basic Bubble Heatmap - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/demos/bubble-heatmap/js/v000.js b/demos/bubble-heatmap/js/v000.js deleted file mode 100644 index 03adb85..0000000 --- a/demos/bubble-heatmap/js/v000.js +++ /dev/null @@ -1,96 +0,0 @@ -function heatmap( selection ) { - var data, - namespace - plot.data = function(_){return arguments.length ? (data = _, plot) : data; }; - plot.namespace = function(_){return arguments.length ? (namespace = _, plot) : namespace; }; - - function plot(){ - var - selections = d3sm.setupStandardChartContainers(selection, namespace, {w: 1000, h:750}, undefined, {axes:{x:0.2, y:0.2},space:{w:1,h:1}}) - container = selections.svg.selection, - bubbles = selections.plot.selection, - xAxis = selections.xAxis.selection, - yAxis = selections.yAxis.selection - - // d3sm.debugQ = true - - - var bhm = d3sm.bubbleHeatmap(bubbles) - .spaceX(selections.plot.rect.w) - .spaceY(selections.plot.rect.h) - .xKey('xKey') - .yKey('yKey') - .vKey('val') - .rKey('val') - .overflowQ(true) - // .minObjectSize(100) - .maxObjectSize(100) - - - // .yKeySortingFunction() - // .xKeySortingFunction(function(a,b){return data[a]['xKey']}) - - .data(data) - - bhm.tooltip().keys(['tooltip']) - bhm() - // - // - // - // // extract keys and values to safe re-computation - var xk = bhm.xValues(), yk = bhm.yValues() - // var pd = [] - // vk.map(function(k, i){ pd.push(...d3.values(vv[k].quartiles)) }) - // pd = d3sm.unique(pd.sort(function (a, b) {return d3.ascending(a, b)})) - // - // console.log(pd) - // console.log(yk, bhm.minObjectSize(), bhm.maxObjectSize(), bhm.ySpacerSize()) - - var yA = d3sm.axis(yAxis) - .spaceX(selections.yAxis.rect.w) - .spaceY(selections.yAxis.rect.h) - .orient("left") - .overflowQ(true) - .categoricalQ(true) - .tickLabels(yk) - .namespace('axis-left') - .minObjectSize(bhm.minObjectSize()) - .maxObjectSize(bhm.maxObjectSize()) - .objectSpacer(bhm.ySpacerSize()) - .objectSize(bhm.ySize()) - - .guideLinesQ(true) - .guidelineSpace(selections.xAxis.rect.w) - yA() - - var xA = d3sm.axis(xAxis) - .spaceX(selections.xAxis.rect.w) - .spaceY(selections.xAxis.rect.h) - .orient("bottom") - .overflowQ(true) - .categoricalQ(true) - .tickLabels(xk) - .namespace('axis-bottom') - .minObjectSize(bhm.minObjectSize()) - .maxObjectSize(bhm.maxObjectSize()) - .objectSpacer(bhm.xSpacerSize()) - .objectSize(bhm.xSize()) - .guideLinesQ(true) - .guidelineSpace(selections.yAxis.rect.h) - - xA() - - console.log(bhm.selection().node(), bhm.selection().node().getBBox(), bhm.selection().node().getBoundingClientRect()) - - - var zoom = d3sm.plotZoom(bhm, xA, yA).eventType('drag') - .orient('2D') - - var pan = d3.zoom().scaleExtent([1,1]).on('zoom', function(){zoom.eventType('drag')()}) - container.call(pan).on('wheel', function(){ zoom.eventType('wheel')() }) - - } - - - return plot -} diff --git a/demos/bundle/index.html b/demos/bundle/index.html deleted file mode 100644 index 3c70c73..0000000 --- a/demos/bundle/index.html +++ /dev/null @@ -1,52 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/demos/filter-select/index.html b/demos/filter-select/index.html deleted file mode 100644 index b289735..0000000 --- a/demos/filter-select/index.html +++ /dev/null @@ -1,155 +0,0 @@ - - - - - Filter Select - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
- - -
-
-
- - - - - - - - - - -
-
- - -
-
Rendered with v0.0.2
- -
-
- -
-
-
-
- - - - - - - - - diff --git a/demos/filter-select/js/v002.txt b/demos/filter-select/js/v002.txt deleted file mode 100644 index e69de29..0000000 diff --git a/demos/filter-table/index.html b/demos/filter-table/index.html deleted file mode 100644 index 1dfaca4..0000000 --- a/demos/filter-table/index.html +++ /dev/null @@ -1,120 +0,0 @@ - - - - - Filter Table - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
- - -
-
Rendered with v0.0.3
- -
-
- -
-
-
-
- - - - - - - - - - diff --git a/demos/filter-table/js/v003.js b/demos/filter-table/js/v003.js deleted file mode 100644 index a32e9c3..0000000 --- a/demos/filter-table/js/v003.js +++ /dev/null @@ -1,222 +0,0 @@ -//VERSION: 0.0.3 -function upset( selection ) { - var - data, - namespace='upset', - barYAxis, - upsetYAxis, - ups, - bar, barR, barRXAxis - - var - upset, - upsetAxis, - setTotalsBarChart, - setTotalsAxis, - intersectionTotalsBarChart, - intersectionTotalsAxis - - plot.data = function(_){return arguments.length ? (data = _, plot) : data; }; - plot.upset = function(_){return arguments.length ? (upset = _, plot) : upset; }; - plot.upsetAxis = function(_){return arguments.length ? (upsetAxis = _, plot) : upsetAxis; }; - plot.setTotalsBarChart = function(_){return arguments.length ? (setTotalsBarChart = _, plot) : setTotalsBarChart; }; - plot.setTotalsAxis = function(_){return arguments.length ? (setTotalsAxis = _, plot) : setTotalsAxis; }; - plot.intersectionTotalsBarChart = function(_){return arguments.length ? (intersectionTotalsBarChart = _, plot) : intersectionTotalsBarChart; }; - plot.intersectionTotalsAxis = function(_){return arguments.length ? (intersectionTotalsAxis = _, plot) : intersectionTotalsAxis; }; - - - - function plot() { - var - svg = d3sm.safeSelect(selection, 'svg'), - - upsetAxisSelection = d3sm.safeSelect(svg, 'g', 'upset-chart-axis'), - setTotalsAxisSelection = d3sm.safeSelect(svg, 'g', 'set-bar-chart-axis'), - intersectionTotalsAxisSelection = d3sm.safeSelect(svg, 'g', 'intersection-bar-chart-axis'), - - upsetSelection = d3sm.safeSelect(svg, 'g', 'upset-chart'), - setTotalsBarChartSelection = d3sm.safeSelect(svg, 'g', 'set-bar-chart'), - intersectionTotalsBarChartSelection = d3sm.safeSelect(svg, 'g', 'intersection-bar-chart') - - - - if (upset == undefined) { upset = d3sm.upset(upsetSelection) } - if (upsetAxis == undefined) { upsetAxis = d3sm.axis(upsetAxisSelection) } - if (setTotalsBarChart == undefined) { setTotalsBarChart = d3sm.bar(setTotalsBarChartSelection) } - if (setTotalsAxis == undefined) { setTotalsAxis = d3sm.axis(setTotalsAxisSelection) } - if (intersectionTotalsBarChart == undefined) { intersectionTotalsBarChart = d3sm.bar(intersectionTotalsBarChartSelection) } - if (intersectionTotalsAxis == undefined) { intersectionTotalsAxis = d3sm.axis(intersectionTotalsAxisSelection) } - - - var - setChartSpacer = 10, - intersectionChartSpacer = 10, - - percentages = { - inChart: {w: 0.75, h: 0.7 }, - usChart: {w: 0.75, h: 0.2 }, - stChart: {w: 0.10, h: 0.2 }, - - inAxis : {w: 0.15, h: 0.7 }, - usAxis : {w: 0.15, h: 0.2 }, - stAxis : {w: 0.10, h: 0.2 }, - - svg: {w: 0.8, h: 0.7} - } - - var - page = {w: window.innerWidth, h: window.innerHeight}, - margins = {left: page.w *0.01, right: page.w *0.01, top: page.h*0.01, bottom: page.h*0.01}, - svgRect = { - _w: page.w * percentages.svg.w, - _h: page.h * percentages.svg.h, - w: page.w * percentages.svg.w - margins.left - margins.right, - h: page.h * percentages.svg.h - margins.top - margins.bottom - }, - chartRect = { - w: svgRect.w - setChartSpacer*2, - h: svgRect.h - intersectionChartSpacer*2 - }, - upsetRect = { - x: margins.left + (chartRect.w * percentages.stChart.w) + (chartRect.w * percentages.usAxis.w) + setChartSpacer*2, - y: margins.top + chartRect.h * percentages.inChart.h + intersectionChartSpacer, - w: chartRect.w * percentages.usChart.w, - h: chartRect.h * percentages.usChart.h - }, - upsetAxisRect = { - x: margins.left + (chartRect.w * percentages.stChart.w) + (chartRect.w * percentages.usAxis.w) + setChartSpacer, - y: margins.top + chartRect.h * percentages.inChart.h + intersectionChartSpacer, - w: chartRect.w * percentages.usAxis.w, - h: chartRect.h * percentages.usAxis.h - }, - setTotalsRect = { - x: margins.left, - y: margins.top + chartRect.h * percentages.inChart.h + intersectionChartSpacer, - w: chartRect.w * percentages.stChart.w, - h: chartRect.h * percentages.stChart.h - }, - setTotalsAxisRect = { - x: margins.left, - y: margins.top + (chartRect.h * percentages.inChart.h) + (chartRect.h * percentages.stChart.h) + intersectionChartSpacer*2, - w: chartRect.w * percentages.stAxis.w, - h: chartRect.h * percentages.stAxis.h - }, - intersectionTotalsRect = { - x: margins.left + (chartRect.w * percentages.stChart.w) + (chartRect.w * percentages.usAxis.w) + setChartSpacer*2, - y: margins.top, - w: chartRect.w * percentages.inChart.w, - h: chartRect.h * percentages.inChart.h - }, - intersectionTotalsAxisRect = { - x: margins.left + (chartRect.w * percentages.stChart.w) + (chartRect.w * percentages.usAxis.w) + setChartSpacer, - y: margins.top, - w: chartRect.w * percentages.inAxis.w, - h: chartRect.h * percentages.inAxis.h - } - - svg.attr('width', svgRect._w).attr('height', svgRect._h) - upsetAxisSelection.attr('transform', 'translate('+upsetAxisRect.x+','+upsetAxisRect.y+')') - setTotalsAxisSelection.attr('transform', 'translate('+setTotalsAxisRect.x+','+setTotalsAxisRect.y+')') - intersectionTotalsAxisSelection.attr('transform', 'translate('+intersectionTotalsAxisRect.x+','+intersectionTotalsAxisRect.y+')') - - upsetSelection.attr('transform', 'translate('+upsetRect.x+','+upsetRect.y+')') - setTotalsBarChartSelection.attr('transform', 'translate('+setTotalsRect.x+','+setTotalsRect.y+')') - intersectionTotalsBarChartSelection.attr('transform', 'translate('+intersectionTotalsRect.x+','+intersectionTotalsRect.y+')') - - - var overflowQ = false - - upset - .data(data) - .spaceX(upsetRect.w) - .spaceY(upsetRect.h) - .xObjectSpacer(0.01) - .yObjectSpacer(0.01) - .maxObjectSize(15) - .minObjectSize(15) - .overflowQ(overflowQ) - // .radius(1) - upset() - - upsetAxis - .spaceX(upsetAxisRect.w) - .spaceY(upsetAxisRect.h) - .orient('left') - .categoricalQ(true) - .tickLabels(upset.setValues()) - .overflowQ(overflowQ) - .minObjectSize(upset.minObjectSize()) - .maxObjectSize(upset.maxObjectSize()) - .objectSpacer(upset.yObjectSpacer()) - - upsetAxis() - - var intersectionTotals = upset.intersectionTotals() - var setTotals = upset.setTotals() - - - intersectionTotalsBarChart.data(intersectionTotals) - .grouping(upset.intersectionValues()) - .spaceX(intersectionTotalsRect.w) - .spaceY(intersectionTotalsRect.h) - .namespace('intersections') - .orient('horizontal') - .valueExtractor(function(k, i){return intersectionTotals[k]['total']}) - .overflowQ(overflowQ) - .minObjectSize(upset.minObjectSize()) - .maxObjectSize(upset.maxObjectSize()) - .objectSpacer(upset.xObjectSpacer()) - - - intersectionTotalsBarChart() - - intersectionTotalsAxis - .spaceX(intersectionTotalsAxisRect.w) - .spaceY(intersectionTotalsAxisRect.h) - .orient('left') - .namespace('intersections-axis-left') - .tickValues(intersectionTotalsBarChart.barValues()) - .guideLinesQ(true) - .guidelineSpace(intersectionTotalsRect.w + intersectionChartSpacer) - .overflowQ(overflowQ) - - intersectionTotalsAxis() - - - - setTotalsBarChart.data(setTotals) - .grouping(upset.setValues()) - .spaceX(setTotalsRect.w) - .spaceY(setTotalsRect.h) - .orient('right') - .valueExtractor(function(k, i){return setTotals[k]['total']}) - .namespace('set') - .overflowQ(overflowQ) - .minObjectSize(upset.minObjectSize()) - .maxObjectSize(upset.maxObjectSize()) - .objectSpacer(upset.yObjectSpacer()) - - - setTotalsBarChart() - - setTotalsAxis - .spaceX(setTotalsAxisRect.w) - .spaceY(setTotalsAxisRect.h) - .namespace('set-axis-bottom') - .orient('bottom') - .tickValues(setTotalsBarChart.barValues()) - .reverseScaleQ(true) - // .tickLabelRotation(0) - .overflowQ(overflowQ) - - setTotalsAxis() - - // var zoom = d3sm.plotZoom(upset, upsetAxis, function(){}).eventType('drag').orient('2D') - // var pan = d3.zoom().scaleExtent([1,1]).on('zoom', function(){ zoom.eventType('drag')() }) - // upsetSelection.call(pan).on('wheel', function(){ zoom.eventType('wheel')() }) - - - } - - return plot -} diff --git a/demos/heatmap/index.html b/demos/heatmap/index.html deleted file mode 100644 index e5c7a19..0000000 --- a/demos/heatmap/index.html +++ /dev/null @@ -1,70 +0,0 @@ - - - - - Basic Heatmap - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/demos/heatmap/js/v000.js b/demos/heatmap/js/v000.js deleted file mode 100644 index 2dff8e0..0000000 --- a/demos/heatmap/js/v000.js +++ /dev/null @@ -1,98 +0,0 @@ -function heatmap( selection ) { - var data, - namespace - plot.data = function(_){return arguments.length ? (data = _, plot) : data; }; - plot.namespace = function(_){return arguments.length ? (namespace = _, plot) : namespace; }; - - function plot(){ - var - selections = d3sm.setupStandardChartContainers(selection, namespace, {w: 1000, h:750}, undefined, {axes:{x:0.2, y:0.2},space:{w:1,h:1}}) - container = selections.svg.selection, - bubbles = selections.plot.selection, - xAxis = selections.xAxis.selection, - yAxis = selections.yAxis.selection - - - d3sm.debugQ = true - - var overflow = true - // var overflow = false - var bhm = d3sm.heatmap(bubbles) - .spaceX(selections.plot.rect.w) - .spaceY(selections.plot.rect.h) - .xKey('xKey') - .yKey('yKey') - .vKey('val') - .overflowQ(overflow) - .objectStrokeWidth(2) - // .minObjectSize(100) - .maxObjectSize(100) - - - // .yKeySortingFunction() - // .xKeySortingFunction(function(a,b){return data[a]['xKey']}) - - .data(data) - - bhm.tooltip().keys(['tooltip']) - bhm() - // - // - // - // // extract keys and values to safe re-computation - var xk = bhm.xValues(), yk = bhm.yValues() - // var pd = [] - // vk.map(function(k, i){ pd.push(...d3.values(vv[k].quartiles)) }) - // pd = d3sm.unique(pd.sort(function (a, b) {return d3.ascending(a, b)})) - // - // console.log(pd) - // console.log(yk, bhm.minObjectSize(), bhm.maxObjectSize(), bhm.ySpacerSize()) - - var yA = d3sm.axis(yAxis) - .spaceX(selections.yAxis.rect.w) - .spaceY(selections.yAxis.rect.h) - .orient("left") - .overflowQ(overflow) - .categoricalQ(true) - .tickLabels(yk) - .namespace('axis-left') - .minObjectSize(bhm.minObjectSize()) - .maxObjectSize(bhm.maxObjectSize()) - .objectSpacer(bhm.ySpacerSize()) - .objectSize(bhm.ySize()) - - .guideLinesQ(true) - .guidelineSpace(selections.xAxis.rect.w) - yA() - - var xA = d3sm.axis(xAxis) - .spaceX(selections.xAxis.rect.w) - .spaceY(selections.xAxis.rect.h) - .orient("bottom") - .overflowQ(overflow) - .categoricalQ(true) - .tickLabels(xk) - .namespace('axis-bottom') - .minObjectSize(bhm.minObjectSize()) - .maxObjectSize(bhm.maxObjectSize()) - .objectSpacer(bhm.xSpacerSize()) - .objectSize(bhm.xSize()) - .guideLinesQ(true) - .guidelineSpace(selections.yAxis.rect.h) - - xA() - - console.log(bhm.selection().node(), bhm.selection().node().getBBox(), bhm.selection().node().getBoundingClientRect()) - - - var zoom = d3sm.plotZoom(bhm, xA, yA).eventType('drag') - .orient('2D') - - var pan = d3.zoom().scaleExtent([1,1]).on('zoom', function(){zoom.eventType('drag')()}) - container.call(pan).on('wheel', function(){ zoom.eventType('wheel')() }) - - } - - - return plot -} diff --git a/demos/heatmap/js/v003.js b/demos/heatmap/js/v003.js deleted file mode 100644 index d8cafa7..0000000 --- a/demos/heatmap/js/v003.js +++ /dev/null @@ -1,132 +0,0 @@ -function heatmap ( selection ) { - var - chart, - xAxis, - yAxis, - legend, - - data, - namespace = 'heatmap', - title = 'Heatmap' - - plot.data = function(_){return arguments.length ? (data = _, plot) : data; }; - plot.title = function(_){return arguments.length ? (title = _, plot) : title; }; - plot.namespace = function(_){return arguments.length ? (namespace = _, plot) : namespace; }; - - plot.chart = function(_){return arguments.length ? (chart = _, plot) : chart; }; - plot.xAxis = function(_){return arguments.length ? (xAxis = _, plot) : xAxis; }; - plot.yAxis = function(_){return arguments.length ? (yAxis = _, plot) : yAxis; }; - plot.legend = function(_){return arguments.length ? (legend = _, plot) : legend; }; - - function plot(){ - var - orient = '2D', - - titleContainer = d3sm.safeSelect(selection, 'h5', d3sm.hypenate(namespace,'title')).text(title), - - selections = d3sm.setupStandardChartContainers( - selection, - namespace, - {w:Math.min(window.innerWidth,1200), h:Math.min(window.innerHeight,700)}, - {top:0.01, bottom:0.01, left:0.01, right:0.01}, - {w:1, h:1}, // percent of container space for svg - {y:0.1,x:0.2}, // percent of container space for axes, - {x:25, margin:05} // absolute width of legendSelect and space on either size - ), - - container = selections.svg.selection, - chartSelection = selections.plot.selection, - xAxisSelection = selections.xAxis.selection, - yAxisSelection = selections.yAxis.selection, - legendSelect = selections.legend.selection - - - if (chart == undefined){ chart = d3sm.heatmap(chartSelection) } - if (yAxis == undefined) { yAxis = d3sm.axis(yAxisSelection) } - if (xAxis == undefined) { xAxis = d3sm.axis(xAxisSelection) } - if (legend == undefined) { - legend = d3sm.numericLegend(legendSelect) - .spaceX(selections.legend.rect.w).spaceY(selections.plot.rect.h) - - } - - // d3sm.debugQ = true - var overflowQ = true - - chart - .spaceX(selections.plot.rect.w) - .spaceY(selections.plot.rect.h) - .xKey('ykey') - .yKey('xkey') - .vKey('padj') - // .minObjectSize(25) - // .maxObjectSize(100) - // .domainPadding(0.01) - .overflowQ(overflowQ) - .data(data) - - chart.tooltip() - .header(function(k, cD){return cD.experimentId}) - .keys(['padj', 'log2FoldChange', 'xkey', 'ykey']) - - chart.colorFunction().colorBy('value') - .valueExtractor(function(k, v, i){ return data[k].padj }) - - chart() - - legend - .min(d3.min(chart.vValues())) - .max(d3.max(chart.vValues())) - .colorFunction(chart.colorFunction()) - .roundTo(5) - legend() - - var xk = chart.xValues(), yk = chart.yValues() - - // console.log(xk, yk) - yAxis - .spaceX(selections.yAxis.rect.w) - .spaceY(selections.yAxis.rect.h) - .orient("left") - .overflowQ(overflowQ) - .categoricalQ(true) - .tickLabels(yk) - .namespace(d3sm.hypenate(namespace,'axis-left')) - .minObjectSize(chart.yMinObjectSize()) - .maxObjectSize(chart.yMaxObjectSize()) - .objectSpacer(chart.objectSpacer()) - .spacerSize(0) - .objectSize(chart.ySize() + chart.ySpacerSize()) - .guideLinesQ(false) - .guidelineSpace(selections.xAxis.rect.w) - - yAxis() - - xAxis = d3sm.axis(xAxisSelection) - .spaceX(selections.xAxis.rect.w) - .spaceY(selections.xAxis.rect.h) - .orient("bottom") - .overflowQ(overflowQ) - .categoricalQ(true) - .tickLabels(xk) - .namespace(d3sm.hypenate(namespace,'axis-bottom')) - .minObjectSize(chart.xMinObjectSize()) - .maxObjectSize(chart.xMaxObjectSize()) - .objectSpacer(chart.objectSpacer()) - .spacerSize(0) - .objectSize(chart.xSize() + chart.xSpacerSize()) - .guideLinesQ(false) - .guidelineSpace(selections.yAxis.rect.h) - xAxis() - - var zoom = d3sm.plotZoom(chart, xAxis, yAxis).eventType('drag') - .orient("vertical") - - var pan = d3.zoom().scaleExtent([1,1]).on('zoom', function(){zoom.eventType('drag')()}) - chartSelection.call(pan).on('wheel', function(){ zoom.eventType('wheel')() }) - - } - - -return plot -} diff --git a/demos/index.html b/demos/index.html index cbe8679..92a645f 100644 --- a/demos/index.html +++ b/demos/index.html @@ -8,43 +8,32 @@ + + + + + + + diff --git a/demos/scatter/index.html b/demos/scatter/index.html deleted file mode 100644 index 539f489..0000000 --- a/demos/scatter/index.html +++ /dev/null @@ -1,141 +0,0 @@ - - - - - Scatter Plot - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
- - -
-
-
- - -
-
Rendered with v0.0.3
- - -
- - - - - - - - - diff --git a/demos/scatter/js/v001.js b/demos/scatter/js/v001.js deleted file mode 100644 index 6d0bea9..0000000 --- a/demos/scatter/js/v001.js +++ /dev/null @@ -1,103 +0,0 @@ -//VERSION: 0.0.1 -function scatterPlot( selection ) { - var data, - namespace - plot.data = function(_){return arguments.length ? (data = _, plot) : data; }; - plot.namespace = function(_){return arguments.length ? (namespace = _, plot) : namespace; }; - - - function plot() { - var - databar = d3sm.safeSelect(selection, 'form', d3sm.hypenate(namespace,'databar')), - - selections = d3sm.setupStandardChartContainers(selection, namespace, {w: 1000, h:500}, undefined, {axes:{x:0.1, y:0.1},space:{w:1,h:1}}) - container = selections.svg.selection, - chart = selections.plot.selection, - xAxis = selections.xAxis.selection, - yAxis = selections.yAxis.selection - - leg = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace,'legend')) - .attr('transform','translate(10, 24)') - - var dataSelect = d3sm.datatoggle(databar) - .keys(d3.keys(data.groupings)) - .namespace(namespace) - .updateFunction(function(){ plot(); zoom.reset(); }) - () - - var currentKey = dataSelect.currentKey() - - var currentData = d3.keys(data.data).reduce( (acc, k) => { - if (data.groupings[currentKey].includes(k)) { acc[k] = data.data[k] } - return acc - }, {}) - - - var scale = currentKey == "linearScale" ? d3.scaleLinear() : d3.scaleSqrt() - var ticks = currentKey == "linearScale" ? 5 : 10 - - var s = d3sm.scatter(chart) - .data(currentData) - .spaceX(selections.plot.rect.w) - .spaceY(selections.plot.rect.h) - .domainPaddingX(0.1) - .domainPaddingY(0.1) - .valueExtractorR(function(d, i){return currentData[d]['r']}) - .maxRadius(20) - .minRadius(1) - .scaleY(scale) - .scaleR(d3.scalePow()) - - var rs = d3.keys(currentData).map(function(d, i){return currentData[d].r}) - - s.colorFunction().colorBy('value').valueExtractor(function(k, v, i){ return v.r }) - .dataExtent( [d3.min(rs), d3.max(rs)]) - - s() - - var xV = s.valuesX(), yV = s.valuesY() - - var yA = d3sm.axis(yAxis) - .spaceX(selections.yAxis.rect.w) - .spaceY(selections.yAxis.rect.h) - .orient("left") - .overflowQ(false) - .tickValues(yV) - .guideLinesQ(true) - .guidelineSpace(selections.xAxis.rect.w) - .namespace('axis-left') - .numberOfTicks(ticks) - .domainPadding(s.domainPaddingY()) - .scale(scale) - yA() - - var xA = d3sm.axis(xAxis) - .spaceX(selections.xAxis.rect.w) - .spaceY(selections.xAxis.rect.h) - .orient("bottom") - .overflowQ(false) - .tickValues(xV) - .guideLinesQ(true) - .guidelineSpace(selections.yAxis.rect.h) - .namespace('axis-bottom') - .numberOfTicks(ticks) - .domainPadding(s.domainPaddingX()) - - xA() - - // set-up zoom, whell and drag - var zoom = d3sm.plotZoom(s, xA, yA).eventType('drag').orient('2D') - var pan = d3.zoom().scaleExtent([1,1]).on('zoom', function(){ zoom.eventType('drag')() }) - container.call(pan).on('wheel', function(){ zoom.eventType('wheel')() }) - - l = d3sm.numericLegend(leg) - .min(d3.min(s.valuesR())) - .max(d3.max(s.valuesR())) - .spaceX(25) - .spaceY(300) - - l() - - } - return plot -} diff --git a/demos/scatter/js/v002.js b/demos/scatter/js/v002.js deleted file mode 100644 index eb27c94..0000000 --- a/demos/scatter/js/v002.js +++ /dev/null @@ -1,117 +0,0 @@ -//VERSION: 0.0.2 -function scatterPlot( selection ) { - var data, - namespace - plot.data = function(_){return arguments.length ? (data = _, plot) : data; }; - plot.namespace = function(_){return arguments.length ? (namespace = _, plot) : namespace; }; - - - function plot() { - var - databar = d3sm.safeSelect(selection, 'form', d3sm.hypenate(namespace,'databar')), - - selections = d3sm.setupStandardChartContainers(selection, namespace, {w: 1000, h:500}, undefined, {axes:{x:0.1, y:0.1},space:{w:1,h:1}}) - container = selections.svg.selection, - chart = selections.plot.selection, - xAxis = selections.xAxis.selection, - yAxis = selections.yAxis.selection - - leg = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace,'legend')) - .attr('transform','translate(10, 24)') - - var dataSelect = d3sm.datatoggle(databar) - .keys(d3.keys(data.groupings)) - .namespace(namespace) - .updateFunction(function(){ plot(); zoom.reset(); }) - () - - var currentKey = dataSelect.currentKey() - - var currentData = d3.keys(data.data).reduce( (acc, k) => { - if (data.groupings[currentKey].includes(k)) { acc[k] = data.data[k] } - return acc - }, {}) - - - var scale = currentKey == "linearScale" ? d3.scaleLinear() : d3.scaleSqrt() - var ticks = currentKey == "linearScale" ? 5 : 10 - - var s = d3sm.scatter(chart) - .data(currentData) - .spaceX(selections.plot.rect.w) - .spaceY(selections.plot.rect.h) - .domainPaddingX(0.1) - .domainPaddingY(0.1) - .valueExtractorR(function(d, i){return currentData[d]['r']}) - .maxRadius(20) - .minRadius(1) - .scaleY(scale) - .scaleR(d3.scalePow()) - - var rs = d3.keys(currentData).map(function(d, i){return currentData[d].r}) - - s.colorFunction().colorBy('value').valueExtractor(function(k, v, i){ return v.r }) - .dataExtent( [d3.min(rs), d3.max(rs)]) - - s() - - var xV = s.valuesX(), yV = s.valuesY() - - var yA = d3sm.axis(yAxis) - .spaceX(selections.yAxis.rect.w) - .spaceY(selections.yAxis.rect.h) - .orient("left") - .overflowQ(false) - .tickValues(yV) - .guideLinesQ(true) - .guidelineSpace(selections.xAxis.rect.w) - .namespace('axis-left') - .numberOfTicks(ticks) - .domainPadding(s.domainPaddingY()) - .scale(scale) - yA() - - var xA = d3sm.axis(xAxis) - .spaceX(selections.xAxis.rect.w) - .spaceY(selections.xAxis.rect.h) - .orient("bottom") - .overflowQ(false) - .tickValues(xV) - .guideLinesQ(true) - .guidelineSpace(selections.yAxis.rect.h) - .namespace('axis-bottom') - .numberOfTicks(ticks) - .domainPadding(s.domainPaddingX()) - - xA() - // - // - // set-up zoom, whell and drag - var zoom = d3sm.plotZoom(s, xA, yA).eventType('drag').orient('2D') - var pan = d3.zoom().scaleExtent([1,1]).on('zoom', function(){ zoom.eventType('drag')() }) - container.call(pan).on('wheel', function(){ zoom.eventType('wheel')() }) - - l = d3sm.numericLegend(leg) - .min(d3.min(s.valuesR())) - .max(d3.max(s.valuesR())) - .spaceX(25) - .spaceY(300) - - l() - - // l = d3sm.categoricLegend(leg) - // .categories(['a','b','c']) - // .spaceX(100) - // .spaceY(300) - // .overflowQ(false) - // .orient('vertical') - // - // l() - - } - - - - - return plot -} diff --git a/demos/scatter/js/v003.js b/demos/scatter/js/v003.js deleted file mode 100644 index b111659..0000000 --- a/demos/scatter/js/v003.js +++ /dev/null @@ -1,187 +0,0 @@ -//VERSION: 0.0.3 -function scatterPlot( selection ) { - var - data, - namespace, - chart, - xAxis, - yAxis, - dataSelect, - legend, - lasso, - lassoWidget - - plot.data = function(_){return arguments.length ? (data = _, plot) : data; }; - plot.namespace = function(_){return arguments.length ? (namespace = _, plot) : namespace; }; - plot.chart = function(_){return arguments.length ? (chart = _, plot) : chart; }; - plot.lasso = function(_){return arguments.length ? (lasso = _, plot) : lasso; }; - - function plot() { - var objectClass= 'item' - - var - lassoDiv = d3sm.safeSelect(selection, 'div', d3sm.hypenate(namespace,'lassoWidget')), - databar = d3sm.safeSelect(selection, 'form', d3sm.hypenate(namespace,'databar')), - selections = d3sm.setupStandardChartContainers( - selection, - namespace, - {w:window.innerWidth, h:window.innerHeight}, - {top:0.01, bottom:0.01, left:0.01, right:0.01}, - {w: 0.8, h:0.8}, // percent of container space for svg - {y:0.1,x:0.1}, // percent of container space for axes, - {x:25, margin:05} // absolute width of legendSelect and space on either size - ), - - // selections = d3sm.setupStandardChartContainers(selection, namespace, {w: 1000, h:500}, undefined, {axes:{x:0.1, y:0.1},space:{w:1,h:1}}) - container = selections.svg.selection, - chartSelection = selections.plot.selection, - xAxisSelection = selections.xAxis.selection, - yAxisSelection = selections.yAxis.selection, - legendSelect = selections.legend.selection - - - var - xAxisToggle = { 'linear': d3.scaleLinear(), 'sqrt': d3.scaleSqrt() }, - yAxisToggle = { 'linear': d3.scaleLinear(), 'sqrt': d3.scaleSqrt() }, - rAxisToggle = { 'linear': d3.scaleLinear(), 'sqrt': d3.scaleSqrt(), 'pow': d3.scalePow() } - - - if (dataSelect == undefined) { dataSelect = d3sm.datatoggle(databar) - .data({ - 'dataset': data.groupings, - 'x-axis': xAxisToggle, - 'y-axis': yAxisToggle, - 'radius scale': rAxisToggle - }) - .namespace(namespace) - .updateFunction(function(){ plot(); zoom.reset(); }) - dataSelect() - } - - if (chart == undefined) { chart = d3sm.scatter(chartSelection) } - if (yAxis == undefined) { yAxis = d3sm.axis(yAxisSelection) } - if (xAxis == undefined) { xAxis = d3sm.axis(xAxisSelection) } - if (legend == undefined) {legend = d3sm.numericLegend(legendSelect) } - - - var currentKeys = dataSelect.currentKeys() - var xScale = xAxisToggle[currentKeys['x-axis']] - var yScale = yAxisToggle[currentKeys['y-axis']] - var rScale = rAxisToggle[currentKeys['radius scale']] - var currentDataKey = currentKeys['dataset'] - - - var currentKey = dataSelect.currentKeys() - var currentData = d3.keys(data.data).reduce( (acc, k) => { - if (data.groupings[currentDataKey].includes(k)) { acc[k] = data.data[k] } - return acc - }, {}) - - - chart - .data(currentData) - .spaceX(selections.plot.rect.w) - .spaceY(selections.plot.rect.h) - .domainPaddingX(0.1) - .domainPaddingY(0.1) - .valueExtractorR(function(d, i){return currentData[d]['r']}) - .maxRadius(20) - .minRadius(1) - .scaleY(yScale) - .scaleX(xScale) - .scaleR(rScale) - .namespace(namespace) - .objectClass(objectClass) - - var rs = d3.keys(currentData).map(function(d, i){return currentData[d].r}) - - chart.colorFunction().colorBy('value').valueExtractor(function(k, v, i){ return v.r }) - .dataExtent( [d3.min(rs), d3.max(rs)]) - chart() - - var xV = chart.valuesX(), yV = chart.valuesY() - - yAxis - .spaceX(selections.yAxis.rect.w) - .spaceY(selections.yAxis.rect.h) - .orient("left") - .overflowQ(false) - .tickValues(yV) - .guideLinesQ(true) - .guidelineSpace(selections.xAxis.rect.w) - .namespace('axis-left') - .numberOfTicks(5) - .domainPadding(chart.domainPaddingY()) - .scale(yScale) - yAxis() - - xAxis = d3sm.axis(xAxisSelection) - .spaceX(selections.xAxis.rect.w) - .spaceY(selections.xAxis.rect.h) - .orient("bottom") - .overflowQ(false) - .tickValues(xV) - .guideLinesQ(true) - .guidelineSpace(selections.yAxis.rect.h) - .namespace('axis-bottom') - .numberOfTicks(5) - .domainPadding(chart.domainPaddingX()) - .scale(xScale) - - xAxis() - - - legend - .min(d3.min(chart.valuesR())) - .max(d3.max(chart.valuesR())) - .spaceX(selections.legend.rect.w) - .spaceY(selections.legend.rect.h) - - legend() - - - - // set-up zoom, whell and drag - var zoom = d3sm.plotZoom(chart, xAxis, yAxis).eventType('drag').orient('2D') - var pan = d3.zoom().scaleExtent([1,1]).on('zoom', function(){ zoom.eventType('drag')() }) - chartSelection.call(pan).on('wheel', function(){ zoom.eventType('wheel')() }) - - - // var svg = selection.select('svg') - // var objs = svg.select('g.'+namespace+'-object-container') - // var objC = '.chart-point' - - // if (lasso == undefined) { lasso = d3sm.lasso( ) - // .svg(svg) - // .objectClass(objC) - // .objectContainer(objs) - // .namespace('test') - // .chartContainer(chartSelection) - // } - // - // lasso - // .xScale(chart.scaleX()) - // .yScale(chart.scaleY()) - // - // lasso() - - - if (lassoWidget == undefined) {lassoWidget = d3sm.lassoWidget(lassoDiv) - .svg(selections.svg.selection) - .objectClass("."+objectClass) - .chartContainer(chartSelection) - .objectContainer(chartSelection.select('g.'+namespace+'-object-container')) - .maxNumberOfGroups(2) - .dataExtractor(function(nodeData){ - return {"name": nodeData} - }) - lassoWidget() - } - - } - - - - - return plot -} diff --git a/demos/tooltip-design/index.html b/demos/tooltip-design/index.html deleted file mode 100644 index 1449187..0000000 --- a/demos/tooltip-design/index.html +++ /dev/null @@ -1,86 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - -
- -
-
-
- Tooltip Title -
- - - - - - - - - - - - - - - - - - - -
NameTooltip
Number of Keys4
Color of valuesCoral
Thiskey has a long value
-
-
- -
- - - - diff --git a/demos/upset/index.html b/demos/upset/index.html deleted file mode 100644 index a9d1c80..0000000 --- a/demos/upset/index.html +++ /dev/null @@ -1,131 +0,0 @@ - - - - - UpSet Plot - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
- - -
-
-
- - -
-
Rendered with v0.0.3
- -
-
- -
-
-
-
- - - - - - - - - - diff --git a/demos/upset/js/v003.js b/demos/upset/js/v003.js deleted file mode 100644 index f08937c..0000000 --- a/demos/upset/js/v003.js +++ /dev/null @@ -1,300 +0,0 @@ -function upset ( selection ) { - var - data, - namespace='upset' - - - var - upset, - upsetAxis, - setTotalsBarChart, - setTotalsAxis, - intersectionTotalsBarChart, - intersectionTotalsAxis, - intersectionSetsAxis - - plot.data = function(_){return arguments.length ? (data = _, plot) : data; }; - plot.namespace = function(_){return arguments.length ? (namespace = _, plot) : namespace; }; - plot.upset = function(_){return arguments.length ? (upset = _, plot) : upset; }; - plot.upsetAxis = function(_){return arguments.length ? (upsetAxis = _, plot) : upsetAxis; }; - plot.setTotalsBarChart = function(_){return arguments.length ? (setTotalsBarChart = _, plot) : setTotalsBarChart; }; - plot.setTotalsAxis = function(_){return arguments.length ? (setTotalsAxis = _, plot) : setTotalsAxis; }; - plot.intersectionTotalsBarChart = function(_){return arguments.length ? (intersectionTotalsBarChart = _, plot) : intersectionTotalsBarChart; }; - plot.intersectionTotalsAxis = function(_){return arguments.length ? (intersectionTotalsAxis = _, plot) : intersectionTotalsAxis; }; - - - - function plot() { - var - svg = d3sm.safeSelect(selection, 'svg'), - - upsetAxisSelection = d3sm.safeSelect(svg, 'g', namespace+'-upset-chart-axis'), - setTotalsAxisSelection = d3sm.safeSelect(svg, 'g', namespace+'-set-bar-chart-axis'), - intersectionTotalsAxisSelection = d3sm.safeSelect(svg, 'g', namespace+'-intersection-bar-chart-axis'), - intersectionSetsAxisSelection = d3sm.safeSelect(svg, 'g', namespace+'-intersection-bar-set-axis'), - - upsetSelection = d3sm.safeSelect(svg, 'g', namespace+'-upset-chart'), - setTotalsBarChartSelection = d3sm.safeSelect(svg, 'g', namespace+'-set-bar-chart'), - intersectionTotalsBarChartSelection = d3sm.safeSelect(svg, 'g', namespace+'-intersection-bar-chart') - - - - if (upset == undefined) { upset = d3sm.upset(upsetSelection) } - if (upsetAxis == undefined) { upsetAxis = d3sm.axis(upsetAxisSelection) } - if (setTotalsBarChart == undefined) { setTotalsBarChart = d3sm.bar(setTotalsBarChartSelection) } - if (setTotalsAxis == undefined) { setTotalsAxis = d3sm.axis(setTotalsAxisSelection) } - if (intersectionTotalsBarChart == undefined) { intersectionTotalsBarChart = d3sm.bar(intersectionTotalsBarChartSelection) } - if (intersectionTotalsAxis == undefined) { intersectionTotalsAxis = d3sm.axis(intersectionTotalsAxisSelection) } - if (intersectionSetsAxis == undefined) { intersectionSetsAxis = d3sm.axis(intersectionSetsAxisSelection) } - - - - var - setChartSpacer = 10, - intersectionChartSpacer = 10, - - percentages = { - inChart: {w: 0.75, h: 0.5 }, - usChart: {w: 0.75, h: 0.2 }, - stChart: {w: 0.10, h: 0.2 }, - - isAxis : {w: 0.75, h: 0.2 }, - inAxis : {w: 0.15, h: 0.5 }, - usAxis : {w: 0.15, h: 0.2 }, - stAxis : {w: 0.10, h: 0.1 }, - - svg: {w: 1, h: 0.7} - } - - var - page = {w: selection.node().getBoundingClientRect().width, h: window.innerHeight}, - margins = {left: page.w *0.01, right: page.w *0.01, top: page.h*0.01, bottom: page.h*0.01}, - svgRect = { - _w: page.w * percentages.svg.w, - _h: page.h * percentages.svg.h, - w: page.w * percentages.svg.w - margins.left - margins.right, - h: page.h * percentages.svg.h - margins.top - margins.bottom - }, - chartRect = { - w: svgRect.w - setChartSpacer*2, - h: svgRect.h - intersectionChartSpacer*2 - }, - upsetRect = { - x: margins.left + (chartRect.w * percentages.stChart.w) + (chartRect.w * percentages.usAxis.w) + setChartSpacer*2, - y: margins.top + chartRect.h * percentages.inChart.h + (chartRect.h * percentages.isAxis.h) + intersectionChartSpacer, - w: chartRect.w * percentages.usChart.w, - h: chartRect.h * percentages.usChart.h - }, - upsetAxisRect = { - x: margins.left + (chartRect.w * percentages.stChart.w) + (chartRect.w * percentages.usAxis.w) + setChartSpacer, - y: margins.top + chartRect.h * percentages.inChart.h + (chartRect.h * percentages.isAxis.h) + intersectionChartSpacer, - w: chartRect.w * percentages.usAxis.w, - h: chartRect.h * percentages.usAxis.h - }, - setTotalsRect = { - x: margins.left, - y: margins.top + chartRect.h * percentages.inChart.h + (chartRect.h * percentages.isAxis.h) + intersectionChartSpacer, - w: chartRect.w * percentages.stChart.w, - h: chartRect.h * percentages.stChart.h - }, - setTotalsAxisRect = { - x: margins.left, - y: margins.top + (chartRect.h * percentages.inChart.h) + (chartRect.h * percentages.stChart.h) + (chartRect.h * percentages.isAxis.h) + intersectionChartSpacer*2, - w: chartRect.w * percentages.stAxis.w, - h: chartRect.h * percentages.stAxis.h - }, - intersectionTotalsRect = { - x: margins.left + (chartRect.w * percentages.stChart.w) + (chartRect.w * percentages.usAxis.w) + setChartSpacer*2, - y: margins.top + (chartRect.h * percentages.isAxis.h), - w: chartRect.w * percentages.inChart.w, - h: chartRect.h * percentages.inChart.h - }, - intersectionTotalsAxisRect = { - x: margins.left + (chartRect.w * percentages.stChart.w) + (chartRect.w * percentages.usAxis.w) + setChartSpacer, - y: margins.top + (chartRect.h * percentages.isAxis.h), - w: chartRect.w * percentages.inAxis.w, - h: chartRect.h * percentages.inAxis.h - }, - intersectionSetsAxisRect = { - x: margins.left + (chartRect.w * percentages.stChart.w) + (chartRect.w * percentages.usAxis.w) + setChartSpacer + intersectionChartSpacer, - y: margins.top + (chartRect.h * percentages.isAxis.h), - w: chartRect.w * percentages.isAxis.w, - h: chartRect.h * percentages.isAxis.h - } - - - svg.attr('width', svgRect._w).attr('height', svgRect._h) - upsetAxisSelection.attr('transform', 'translate('+upsetAxisRect.x+','+upsetAxisRect.y+')') - setTotalsAxisSelection.attr('transform', 'translate('+setTotalsAxisRect.x+','+setTotalsAxisRect.y+')') - intersectionTotalsAxisSelection.attr('transform', 'translate('+intersectionTotalsAxisRect.x+','+intersectionTotalsAxisRect.y+')') - intersectionSetsAxisSelection.attr('transform', 'translate('+intersectionSetsAxisRect.x+','+intersectionSetsAxisRect.y+')') - - upsetSelection.attr('transform', 'translate('+upsetRect.x+','+upsetRect.y+')') - setTotalsBarChartSelection.attr('transform', 'translate('+setTotalsRect.x+','+setTotalsRect.y+')') - intersectionTotalsBarChartSelection.attr('transform', 'translate('+intersectionTotalsRect.x+','+intersectionTotalsRect.y+')') - - var overflowQ = true - - upset - .data(data) - .spaceX(upsetRect.w) - .spaceY(upsetRect.h) - .xObjectSpacer(0.01) - .yObjectSpacer(0.01) - .maxObjectSize(5) - .minObjectSize(20) - .overflowQ(overflowQ) - .intersectionExtractor(function(k, i){ return data[k].intersection.join(';') }) - .intersectionKeySortingFunction(function(a, b){ - return ( - data[a].intersection.length - data[b].intersection.length - || - upset.intersectionValues().indexOf(data[a].intersection.join(';')) - -upset.intersectionValues().indexOf(data[b].intersection.join(';')) - ) - }) - // .radius(1) - upset() - - upsetAxis - .spaceX(upsetAxisRect.w) - .spaceY(upsetAxisRect.h) - .orient('left') - .categoricalQ(true) - .tickLabels(upset.setValues()) - .overflowQ(overflowQ) - .minObjectSize(upset.minObjectSize()) - .maxObjectSize(upset.maxObjectSize()) - .objectSpacer(upset.yObjectSpacer()) - .tickLabelMargin(30) - - upsetAxis() - - var intersectionTotals = upset.intersectionTotals() - var setTotals = upset.setTotals() - - - intersectionSetsAxis - .tickLabels(upset.intersectionValues()) - .categoricalQ(true) - .spaceX(intersectionSetsAxisRect.w) - .spaceY(intersectionSetsAxisRect.h) - .orient('top') - .namespace('intersection-sets-name') - .overflowQ(overflowQ) - .minObjectSize(upset.minObjectSize()) - .maxObjectSize(upset.maxObjectSize()) - .objectSpacer(upset.xObjectSpacer()) - .objectSize(upset.xObjectSize()) - .spacerSize(upset.xSpacerSize()) - // .tickLabelRotation(-45) - .tickTickLabelSpacer(0) - - intersectionSetsAxis() - - intersectionTotalsBarChart.data(intersectionTotals) - .grouping(upset.intersectionValues()) - .spaceX(intersectionTotalsRect.w) - .spaceY(intersectionTotalsRect.h) - .namespace('intersections') - .orient('horizontal') - .valueExtractor(function(k, i){return intersectionTotals[k]['total']}) - .overflowQ(overflowQ) - .minObjectSize(upset.minObjectSize()) - .maxObjectSize(upset.maxObjectSize()) - .objectSpacer(upset.xObjectSpacer()) - .objectSize(upset.xObjectSize()) - .spacerSize(upset.xSpacerSize()) - - - intersectionTotalsBarChart.tooltip().keys(['Amount','Values']) - .values([ - function(currentData, d){ - return currentData.total - }, - function(currentData, d){ - var l = currentData.values.length - return currentData.values.join(', ').slice(0,100)+ ((l > 100) ? '...': '') - } - ]) - - // intersectionTotalsBarChart.colorFunction().colors(['cyan', 'blue']).colorBy('value') - - // intersectionSetsAxis() - intersectionTotalsBarChart() - - intersectionTotalsAxis - .spaceX(intersectionTotalsAxisRect.w) - .spaceY(intersectionTotalsAxisRect.h) - .orient('left') - .namespace('intersections-axis-left') - .tickValues(intersectionTotalsBarChart.barValues()) - .guideLinesQ(true) - .guidelineSpace(intersectionTotalsRect.w + intersectionChartSpacer) - .overflowQ(overflowQ) - - intersectionTotalsAxis() - - - - setTotalsBarChart.data(setTotals) - .grouping(upset.setValues()) - .spaceX(setTotalsRect.w) - .spaceY(setTotalsRect.h) - .orient('right') - .valueExtractor(function(k, i){return setTotals[k]['total']}) - .namespace('set') - .overflowQ(overflowQ) - .minObjectSize(upset.minObjectSize()) - .maxObjectSize(upset.maxObjectSize()) - .objectSpacer(upset.yObjectSpacer()) - .objectSize(upset.yObjectSize()) - .spacerSize(upset.ySpacerSize()) - .barPercent(0.75) - - setTotalsBarChart() - - - - setTotalsAxis - .spaceX(setTotalsAxisRect.w) - .spaceY(setTotalsAxisRect.h) - .namespace('set-axis-bottom') - .orient('bottom') - .tickValues(setTotalsBarChart.barValues()) - .reverseScaleQ(true) - // .tickLabelRotation(0) - .overflowQ(overflowQ) - - setTotalsAxis() - - - - - - - var zoom = d3sm.multiPlotZoom(upset).eventType('drag').orient('2D') - .xComponents([ - intersectionTotalsBarChart, - intersectionSetsAxis - ]) - .yComponents([ - upsetAxis, - setTotalsBarChart - ]) - var pan = d3.zoom().scaleExtent([1,1]).on('zoom', function(){ zoom.eventType('drag')() }) - upsetSelection.call(pan).on('wheel', function(){ zoom.eventType('wheel')() }) - intersectionTotalsBarChartSelection.call(pan).on('wheel', function(){ zoom.eventType('wheel')() }) - - - - intersectionTotalsBarChartSelection.selectAll('.bar.intersections > rect.bar-rect').on('click', function(d, i){ - console.log( d, i) - }) - // selection.selectAll('text').attr('pointer-events', 'none') - // .attr('pointer-events', 'none') - - } - - return plot -} diff --git a/demos/vertical-violins/index.html b/demos/vertical-violins/index.html deleted file mode 100644 index 4020970..0000000 --- a/demos/vertical-violins/index.html +++ /dev/null @@ -1,147 +0,0 @@ - - - - - Basic Violins - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/dist/d3sm.esm.js b/dist/d3sm.esm.js index b944a52..b9ee62a 100644 --- a/dist/d3sm.esm.js +++ b/dist/d3sm.esm.js @@ -1,11 +1,3 @@ -// import {hasQ} from './array-functions'; -/******************************************************************************* -** ** -** ** -** HELPERS ** -** ** -** ** -*******************************************************************************/ /** * Helper function for Array.filter to get unique elements of the array * @param {*} value current value as mapping over array (self) @@ -16,270 +8,175 @@ function uniqueElements(value, index, self) { return self.indexOf(value) === index; } /** -* Extracts x and y of translate from transform property -* @param {string} transform transform property of svg element -* @returns {number[]} x, y of translate(x, y) +* This function tests to see if all elements of the passed array are true. +* @param {Array} array of values +* @param {Function} [func function(value){return value == true;}] is applied to each value of the array and should return a boolean. +* @returns {boolean} if all values are true by function */ -function getTranslation(transform) { - // Create a dummy g for calculation purposes only. This will never - // be appended to the DOM and will be discarded once this function - // returns. - var g = document.createElementNS('http://www.w3.org/2000/svg', 'g'); - // Set the transform attribute to the provided string value. - transform = transform == undefined ? 'translate(0,0)' : transform; - g.setAttributeNS(null, 'transform', transform); - // consolidate the SVGTransformList containing all transformations - // to a single SVGTransform of type SVG_TRANSFORM_MATRIX and get - // its SVGMatrix. - var matrix = g.transform.baseVal.consolidate().matrix; - // As per definition values e and f are the ones for the translation. - return [matrix.e, matrix.f]; +function all( array, func ) { + if (func == undefined) { return array.every( function(value) { return value === true; }); } + return array.every( function(value) { return func(value); } ); } - /** -* Modifies luminance of hexidecimal number -* @param {string} hex should be hexidecimal value with or without the proceeding octotrope -* @param {number} lum value to increase or decrease luminosity by -* @returns {string} updated hexidecimal value without the proceeding octotrope +* Counts the number of occurances of each element in the given array. +* @param {Array} array of elements +* @returns {Object} of key: value pairs where key is an element in the array and value is the number of times it occurs. */ -function modifyHexidecimalColorLuminance(hex, lum) { - // validate hex string - var hex = String(hex).replace(/[^0-9a-f]/gi, ''); - - if (hex.length < 6) { - hex = hex[0]+hex[0]+hex[1]+hex[1]+hex[2]+hex[2]; - } - lum = lum || 0; - - // convert to decimal and change luminosity - var rgb = '#', c, i; - for (i = 0; i < 3; i++) { - c = parseInt(hex.substr(i*2,2), 16); - c = Math.round(Math.min(Math.max(0, c + (c * lum)), 255)).toString(16); - rgb += ('00'+c).substr(c.length); - } - - return rgb; +function tally( array ) { + var tallies = {}; + array.map( function ( element ) { + if ( hasQ(Object.keys(tallies), element) ) { tallies[element] = 1; } + else { tallies[element] += 1; } + }); + return tallies; } - - /** -* Calculated the quartiles of the passed data and stores them with qKeys -* @param {number[]} data list of numerical values -* @param {string[]} [qKeys=['q0', 'q1', 'q2', 'q3', 'q4']] how returned object with quartiles should be stored -* @returns {Object} with keys qKeys giving only the numerical values for the quartiles +* Short-hand for array.includes(item); +* @param {Array} array +* @param {*} item to test if contained in {array} +* @returns {boolean} */ -function quartiles(data, qKeys) { - var - q2 = d3.median(data), - lower = data.filter(x => x < q2), - upper = data.filter(x => x > q2), - - q1 = d3.median(lower), - q1 = q1 == undefined ? q2 : q1, - - q0 = d3.min(lower), - q0 = q0 == undefined ? q1 : q0, - - q3 = d3.median(upper), - q3 = q3 == undefined ? q2 : q3, - - q4 = d3.max(upper), - q4 = q4 == undefined ? q3 : q4, - - k0 = 'q0', k1 = 'q1', k2 = 'q2', k3 = 'q3', k4 = 'q4', - obj = {}; - if (qKeys!=undefined && qKeys.length == 5) { k0 = qKeys[0]; k1 = qKeys[1]; k2 = qKeys[2]; k3 = qKeys[3]; k4 = qKeys[4]; } - obj[k0] = q0; obj[k1] = q1; obj[k2] = q2; obj[k3] = q3; obj[k4] = q4; - - return obj -} - +function hasQ( array, item ) { return array.includes(item); } /** -* Helper function to get all values needed in making violin plots -* @param {string[]} violinKeys -* @param {number[]} data -* @param {Function} valueExtractorFunction how to get values from data[violinKeys[i]] -* @param {boolean} horizontalQ whether or not violins will be rendered horizontally or vertically -* @param {string} qKey how the object containing the quartiles should be labeled as -* @param {string[]} qKeys how each quartile should be labeled as -* @returns {Object} required for @see{@link violin} containing keys values, binnned, frequencies, points, and quartiles -* @see{@link quartiles} +* Returns first item in array +* @param {Array} array of items +* @returns {*} array[0] */ -function extractViolinValues( - violinKeys, - data, - valueExtractorFunction, - horizontalQ, - qKey, - qKeys -){ - var obj = {}; - violinKeys.map(function(k, i){ - var d = valueExtractorFunction(k, i, data), - binned = d3.histogram()(d), - frequencies = binned.map(x=>x.length), - minPoint = horizontalQ ? {y: d3.min(d), x: 0} : {x: d3.min(d), y: 0}, - maxPoint = horizontalQ ? {y: d3.max(d), x: 0} : {x: d3.max(d), y: 0}, - points = binned.map(function(bin, i) { - return horizontalQ - ? {y: (bin.length) ? d3.median(bin): d3.median([bin.x0, bin.x1]), x: frequencies[i]} - : {x: (bin.length) ? d3.median(bin): d3.median([bin.x0, bin.x1]), y: frequencies[i]} - }), - quarts = quartiles(d, qKeys), - o = { - values: d, - binned: binned, - frequencies: frequencies, - points: [minPoint].concat(points).concat([maxPoint]) - }; - o[qKey] = quarts; - obj[k] = o; - }); - return obj; -} +function first( array ) { return array[0]; } /** -* Hypenates all strings together -* @param {string[]} arguments -* @returns {string} "arg1-arg2-...-argn" +* Returns last item in array +* @param {Array} array of items +* @returns {*} array[array.length-1] */ -function hypenate(){ return Array.prototype.slice.call(arguments).join('-') } - +function last( array ) { return array[array.length-1]; } /** -* Rounds decimals of number to precision -* @param {number} number -* @param {number} precision -* @returns {number} rounded to precision +* Calculates the total value of numbers in passed array +* @param {number[]} array of numerical values +* @returns {number} sum over elements in array */ -function round(number, precision) { - var shift = function (number, precision, reverseShift) { - if (reverseShift) { - precision = -precision; - } - var numArray = ('' + number).split('e'); - return +(numArray[0] + 'e' + (numArray[1] ? (+numArray[1] + precision) : precision)); - }; - return shift(Math.round(shift(number, precision, false)), precision, true); -} - +function total$1( array ) { return array.reduce((a, b) => a + b, 0) } /** -* recursively ascends element.parentElement to find a svg tag -* @param {Element} element -* @returns {Element | undefined} +* Removes duplicates in array +* @param {Array} array of items +* @returns {Array} of items such that item_i != item_j for all i < j +* @see{@link uniqueElements} for the filtering function */ -function getContainingSVG(element) { - var parent = element.parentElement; - var tag = parent.tagName.toLowerCase(); - if (tag === 'svg') { return parent; } - if (tag === 'html') { return undefined; } - return getContainingSVG(parent); -} +function unique( array ) { return array.filter( uniqueElements ); } /** -* Maps arguments in to d3.interpolateRgbBasis -* @param arguments -* @returns {Function} +* Filters passed array for specified indicies +* @param {Array} array of items +* @param {number[]} positions of integers such that i < array.length +* @returns {Array} of items such that for any item_i, positions.includes(i) === true */ -function interpolateColors(){return d3.interpolateRgbBasis(arguments)} - +function get( array, positions ) { + return array.filter( function( value, index ) { return hasQ(positions, index); } ); +} /** -* Trys to reduce text to fit in specified area, made for tick labels as called by -* @see{@link axis} -* @param {d3.selection} t container for specific axis tick -* @param {string} text to be the label of the passed axis tick -* @param {boolean} orient of the axis, true is horizontal, false is vertical -* @param {number} tickLength is the length of the text -* @param {number} space is the amount of availble space for the text and the tick to fit in -* @param {boolean} overflowQ whether or not allowed to go over the alloted space -* @returns {none} +* Determines if all elements in passed array are arrays themselves. +* @param {Array} array of items +* @returns {boolean} true if Array.isArray(e) is true for all e in array +* @see{@link all} */ -function truncateText(t, text, orient, tickLength, space, overflowQ) { - var rect = t.node().getBoundingClientRect(); - t.text(text); - while (Math.max(rect.width, rect.height) > space - tickLength) { - text = String(text); - text = text.slice(0, text.length - 1); - t.text(text + '...'); - rect = t.node().getBoundingClientRect(); - if (text.length == 0) break - } -} - -function truncateString(string, space, font) { - var chars = space / font; - if (chars < string.length) { - return string.slice(0, Math.round(chars-5)) + '...' - } else { - return string - } +function listOfListsQ( array ) { + return all( array.map( function( element, index ) { return Array.isArray(element) } ) ) } - /** -* Trys to use d3.selection to get element, if it doesnt exist, makes one -* @param {d3.selection} sel selection in which to try and find object -* @param {string} tag tag of which to try and select -* @param {string} [cls=''] class of tag to try and grab -* @returns {d3.selection} of either append or selected tag.cls within sel +* Built on top of @see{@link get}, mapping if positions is a list of lists (@see{@link listOfListsQ}) +* @param {Array} array of items +* @param {number[] | []number[] } positions of integers or list of positions of integers +* @returns {boolean} returns specified positions from array. If nested positions passed, returns requested items in same structure. */ -function safeSelect(sel, tag, cls) { - var clsStr = cls == undefined ? '' : '.'+cls; - var sSel = sel.select(tag+clsStr).empty() - ? sel.append(tag) - : sel.select(tag+clsStr); - return sSel - .classed(clsStr.replace('.', ''), true) - .attr('transform', sSel.attr('transform') == undefined ? 'translate(0,0)' : sSel.attr('transform')) +function cut( array, positions ) { + if ( listOfListsQ(array) ) { return positions.map(function(pos, i) { return array.get(pos); }); } + return get( array, positions ); } /** -* evenly partitions the range [min, max] into n parts -* @param {number} min -* @param {number} max -* @param {number} n -* @returns {number[]} array of length n evenly partitioned between min and max +* Given an array of objects, constructs new objects where each value is a list +* based on the corresonding key, which is extracted by the parameter by +* @param {Objects[]} array of objects +* @param {string} by key within all objects of passed array +* @param {string[]} [groups] saves some computation if all known values extracted by mapping over the parameter by are passed +* @returns {Object} of key value pairs, where keys are all values of the key by from an object in the passed array and the value are those corresponding objects. */ -function tickRange(min, max, n) { - var a = [min]; - var d = max-min; - var s = d / (n-1); - for (var i = 0; i < n-2; i++) { a.push(min + s * (i+1)); } - a.push(max); - return a -} - +function groupBy (array, by, groups) { + if (groups == undefined) { + groups = unique(array.map(function(elements, index){ return element[by]; })); + groups.map(function(value, index){groupped[value] = [];}); + } -function euclideanDistance(p1, p2){ - var a = p1[0] - p2[0], b = p1[1] - p2[1]; - return Math.sqrt(a*a + b*b) + var groupped = {}; + array.map(function(element, index){groupped[element[by]].push(element);}); + return groupped } /** -* Short-hand for array.includes(item); +* Tests if two arrays are equivalent * @param {Array} array -* @param {*} item to test if contained in {array} -* @returns {boolean} +* @param {Array} other +* @returns {boolean} if every element of array matches that of other */ -function hasQ( array, item ) { return array.includes(item); } +function arrayEquals(array, other) { + if (!other) + return false; + // compare lengths - can save a lot of time + if (array.length != other.length) + return false; + + for (var i = 0, l=array.length; i < l; i++) { + // Check if we have nested arrays + if (array[i] instanceof Array && other[i] instanceof Array) { + // recurse into the nested arrays + if (!arrayEquals(array[i],other[i])) + return false; + } + else if (array[i] != other[i]) { + // Warning - two different object instances will never be equal: {x:20} != {x:20} + return false; + } + } + return true; +} + + /** -* Calculates the total value of numbers in passed array -* @param {number[]} array of numerical values -* @returns {number} sum over elements in array +* Recursively tallies the number of elements at each level of the passed, putatively nested array +* @param {Array} array of items which may include nested arrays +* @param {number} [level=0] current depth in the recursion +* @param {Array} [levelData=[]] keeps track of items seen so far at each depth +* @returns {Array} stating the number of elements (array inclusive) found at each level of the array */ -function total( array ) { return array.reduce((a, b) => a + b, 0) } +function elementsAtLevels(array, level, levelData) { + level = level == undefined ? 0 : level + 1; + levelData = levelData == undefined ? [] : levelData; + if ( level >= levelData.length ) { levelData.push(array.length);} else {levelData[level] += array.length; } + array.map(function(e, i) {if (Array.isArray(e)){ elementsAtLevels(e, level, levelData); }}); + return levelData +} + + /** -* Removes duplicates in array -* @param {Array} array of items -* @returns {Array} of items such that item_i != item_j for all i < j -* @see{@link uniqueElements} for the filtering function +* Recursively tallies the number of elements of the passed, putatively nested array +* @param {Array} array of items which may include nested arrays +* @param {number} [elements=0] current number of elements seen so far +* @returns {number} number of elements (array inclusive) found in passed array */ -function unique( array ) { return array.filter( uniqueElements ); } +function numberOfElements( array, elements ) { + elements = elements == undefined ? 0 : elements; + array.map(function(e, i) { + if ( Array.isArray(e) ) { elements = numberOfElements(e, elements); } + else { elements += 1; } + }); + return elements +} /** * Concats all nested arrays in passed array to form a single array @@ -308,258 +205,255 @@ function whichBin(bins, value) { return i } +var arr = /*#__PURE__*/Object.freeze({ + uniqueElements: uniqueElements, + all: all, + tally: tally, + hasQ: hasQ, + first: first, + last: last, + total: total$1, + unique: unique, + get: get, + listOfListsQ: listOfListsQ, + cut: cut, + groupBy: groupBy, + arrayEquals: arrayEquals, + elementsAtLevels: elementsAtLevels, + numberOfElements: numberOfElements, + flatten: flatten, + whichBin: whichBin +}); + /** - * calls console.group if d3sm.debugQ == true - * @param {string} name of the group - * @returns {undefined} - */ -function consoleGroup(name) { - if (window.d3sm.debugQ === true){ - console.group(name); - } +* Modifies luminance of hexidecimal number +* @param {string} hex should be hexidecimal value with or without the proceeding octotrope +* @param {number} lum value to increase or decrease luminosity by +* @returns {string} updated hexidecimal value without the proceeding octotrope +*/ +function modifyHexidecimalColorLuminance(hex, lum) { + // validate hex string + var hex = String(hex).replace(/[^0-9a-f]/gi, ''); + + if (hex.length < 6) { + hex = hex[0]+hex[0]+hex[1]+hex[1]+hex[2]+hex[2]; + } + lum = lum || 0; + + // convert to decimal and change luminosity + var rgb = '#', c, i; + for (i = 0; i < 3; i++) { + c = parseInt(hex.substr(i*2,2), 16); + c = Math.round(Math.min(Math.max(0, c + (c * lum)), 255)).toString(16); + rgb += ('00'+c).substr(c.length); + } + + return rgb; } /** - * calls console.groupEnd if d3sm.debugQ == true - * @returns {undefined} - */ -function consoleGroupEnd() { - if (window.d3sm.debugQ === true){ - console.groupEnd(); - } -} +* Maps arguments in to d3.interpolateRgbBasis +* @param arguments +* @returns {Function} +*/ +function interpolateColors(){return d3.interpolateRgbBasis(arguments)} + +var color = /*#__PURE__*/Object.freeze({ + modifyHexidecimalColorLuminance: modifyHexidecimalColorLuminance, + interpolateColors: interpolateColors +}); /** - * Calls console.log if d3sm.debugQ == true - * @param {string} func name of the function logging - * @param {string} msg to log - * @param {Object} data to be logged along side the message - * @returns {undefined} - */ -function log(func, msg, data) { - if (window.d3sm.debugQ === true){ - console.log( - `%c[d3sm::${func}]:\t${msg}`, - [ - 'background: #6cd1ef', - 'border-radius: 5000px', - 'padding: 0px 2px', - 'font-size: 14px' - ].join(';') - ); - console.table(data); - // console.trace() - } +* Rounds decimals of number to precision +* @param {number} number +* @param {number} precision +* @returns {number} rounded to precision +*/ +function round(number, precision) { + var shift = function (number, precision, reverseShift) { + if (reverseShift) { + precision = -precision; + } + var numArray = ('' + number).split('e'); + return +(numArray[0] + 'e' + (numArray[1] ? (+numArray[1] + precision) : precision)); + }; + return shift(Math.round(shift(number, precision, false)), precision, true); } /** - * Calls console.warn if d3sm.debugQ == true - * @param {string} func name of the function warning - * @param {string} msg to display - * @param {Object} data to be displayed along side the message - * @returns {undefined} - */ -function warn(func, msg, data) { - if (window.d3sm.debugQ === true) - console.warn( - `%c[d3sm::${func}]:\t${msg}`, - [ - 'background: #ffd53e', - 'border-radius: 5000px', - 'padding: 0px 2px', - 'font-size: 14px' - ].join(';') - ); - console.table(data); +* evenly partitions the range [min, max] into n parts +* @param {number} min +* @param {number} max +* @param {number} n +* @returns {number[]} array of length n evenly partitioned between min and max +*/ +function tickRange(min, max, n) { + var a = [min]; + var d = max-min; + var s = d / (n-1); + for (var i = 0; i < n-2; i++) { a.push(min + s * (i+1)); } + a.push(max); + return a } + /** - * Calls the console.info if d3sm.debugQ == true - * @param {string} func name of the function providing info - * @param {string} msg to display - * @param {Object} data to be displayed along side the message - * @returns {undefined} - */ -function info(func, msg, data) { - if (window.d3sm.debugQ) - console.info( - `%c[d3sm::${func}]:\t${msg}`, - [ - 'background: #009ccd', - 'border-radius: 5000px', - 'padding: 0px 2px', - 'font-size: 14px' - ].join(';') - ); - console.table(data); +* @deprecated @see{@link tickRange} +* @param {number} min +* @param {number} max +* @param {number} parts +* @returns {number[]} array of length parts evenly partitioned between min and max +*/ +function partitionRangeInto(min, max, parts) { + var diff = max - min; + return Array(parts).map(function (e, i) { return min + diff / parts * i }) } +function euclideanDistance(p1, p2){ + var a = p1[0] - p2[0], b = p1[1] - p2[1]; + return Math.sqrt(a*a + b*b) +} + /** - * Calls console.error if d3sm.debugQ == true - * @param {string} func name of the function which sends the error - * @param {string} msg to display - * @param {Object} data to be displayed along side the message - * @returns {undefined} - */ -function error(func, msg, data) { - if (window.d3sm.debugQ) - console.error(`[d3sm::${func}]:\t${msg}\t%o`,data); +* Calculated the quartiles of the passed data and stores them with qKeys +* @param {number[]} data list of numerical values +* @param {string[]} [qKeys=['q0', 'q1', 'q2', 'q3', 'q4']] how returned object with quartiles should be stored +* @returns {Object} with keys qKeys giving only the numerical values for the quartiles +*/ +function quartiles(data, qKeys) { + var + q2 = d3.median(data), + lower = data.filter(x => x < q2), + upper = data.filter(x => x > q2), + + q1 = d3.median(lower), + q1 = q1 == undefined ? q2 : q1, + + q0 = d3.min(lower), + q0 = q0 == undefined ? q1 : q0, + + q3 = d3.median(upper), + q3 = q3 == undefined ? q2 : q3, + + q4 = d3.max(upper), + q4 = q4 == undefined ? q3 : q4, + + k0 = 'q0', k1 = 'q1', k2 = 'q2', k3 = 'q3', k4 = 'q4', + obj = {}; + if (qKeys!=undefined && qKeys.length == 5) { k0 = qKeys[0]; k1 = qKeys[1]; k2 = qKeys[2]; k3 = qKeys[3]; k4 = qKeys[4]; } + obj[k0] = q0; obj[k1] = q1; obj[k2] = q2; obj[k3] = q3; obj[k4] = q4; + + return obj } +/** +* determines the width of an object for the calling plotting function +* @param {number} freeSpace how much space is avalible +* @param {number} numberOfObjects how many object do we need +* @param {number} minObjectWidth how small are these objects allowed to be +* @param {number} maxObjectWidth how large are these object allowed to be +* @param {number} sizeOfSpacer percent of freeSpace that a single spacer should take up (need numberOfObjects - 1 spacers) +* @param {boolean} overflowQ can we go beyond alloted space +* @returns {number} how large object should be +* function tries to keep object within min / max width, but wil default to +* 5e-10 (smallest consistenly visible by svg size of element) if overflowQ is false +*/ +function calculateWidthOfObject(freeSpace, numberOfObjects, minObjectWidth, maxObjectWidth, sizeOfSpacer, overflowQ) { + var sizeOfSpacer = + sizeOfSpacer == 0 || sizeOfSpacer > 1 + ? sizeOfSpacer + : freeSpace * sizeOfSpacer; + var numberOfSpacers = numberOfObjects - 1; + var spaceTakenBySpacers = numberOfSpacers * sizeOfSpacer; + var remainingSpace = freeSpace - spaceTakenBySpacers; + remainingSpace = remainingSpace < 0 ? 0 : remainingSpace; + var objectWidth = remainingSpace / numberOfObjects; + if ( overflowQ && minObjectWidth != undefined && objectWidth < minObjectWidth ) { objectWidth = minObjectWidth; } + // if ( maxObjectWidth != undefined && objectWidth > maxObjectWidth ) { objectWidth = maxObjectWidth } + if ( overflowQ && maxObjectWidth != undefined && objectWidth < maxObjectWidth ) { objectWidth = maxObjectWidth; } + return Math.max(objectWidth, 5e-10) +} /** -* Function for setting up containers for most plots with the y axis container -* positioned on the left and the x axis container positioned on the bottom -* @param {d3.selection} selection selection of container in which the svg is or should be made -* @param {string} namespace namespace of the chart -* @param {Object} [space={w:window.innerWidth, h:window.innerHeight}] the width (w) and height (h) availble -* @param {number} [space.w=window.innerWidth] the available width in which to render the chart -* @param {number} [space.h=window.innerHeight] the available height in which to render the chart - -* @param {Object} [margins={top: 0.01, bottom: 0.01, left: 0.01, right: 0.01}] the margins for the chart -* @param {number} [margins.top=0.01] the top margin of the chart -* @param {number} [margins.bottom=0.01] the bottom margin of the chart -* @param {number} [margins.left=0.01] the left margin of the chart -* @param {number} [margins.right=0.01] the right margin of the chart - - -* @param {Object} [percentages = {axes:{x:0.1,y:0.1},space:{w:0.8,h:0.6}}] percentages of the paramater space of which to make the x and y axes as well as the percent of the availble space in which to render the plot -* @param {Object} [percentages.axes={x:0.1,y:0.1}] the percentages of the paramater space, of which the x and y axes will take up -* @param {number} [percentages.axes.xAxisPercent=0.1] the percentages of the paramater space, of which the x axis will take up -* @param {number} [percentages.axes.yAxisPercent=0.1] the percentages of the paramater space, of which the y axis will take up - -* @param {Object} [percentages.space={w:0.8,h:0.6}] the percentages of the paramater space, of which the SVG's width and height will be set -* @param {number} [percentages.space.percentOfSpaceForWidth=0.1] the percentages of the paramater space, of which the SVG's width will be set -* @param {number} [percentages.space.percentOfSpaceForHeight=0.1] the percentages of the paramater space, of which the SVG's height will be set - -* @returns {Object} returns the selection and "boundingRects" of the plot container, x-axis container and y-axis container -* as -* -* { -* -* plot: {selection: plotSelection, rect: plotRect}, -* -* xAxis:{selection:xAxisSelection, rect:xAxisRect}, -* -* yAxis: {selection:yAxisSelection, rect:yAxisRect} -* -* } -* -* where each rect has form: -* -* {x: #, y: #, h: #, w: #} -* -* depicting the starting x and y coordinate of the coresponding container (also their default transform values) as well their height (h) ans width (w) +* @param {Array[]} data list data (can be nested). If nested will create more complex spacer size +* @param {number} freeSpace how much space is avalible +* @param {number} objectWidth @see{@link calculateWidthOfObject} +* @param {number} numberOfObjects how many object do we need +* @param {number} baseSpacerSize percent of freeSpace that a single spacer should take up (need numberOfObjects - 1 spacers) +* @param {boolean} overflowQ can we go beyond alloted space +* @returns {number} returns size that spacer should be at level=0 */ -// export function setupStandardChartContainers( selection, namespace, space, margins, percentages) { -// export function setupStandardChartContainers( -// selection, -// namespace, -// space={w:availableWidth=window.innerWidth, h:availableHeight=window.innerHeight}, -// margins={top:0.01, bottom:0.01, left:0.01, right:0.01}, -// percentages={axes: {x: xAxisPercent=0.1, y: yAxisPercent=0.1}, space: {w: percentOfSpaceForWidth, h: percentOfSpaceForHeight}} -// ) { -// if (space == undefined) { space = {w: window.innerWidth, h: window.innerHeight} } -// if (margins == undefined) { margins = {top: 0.01, bottom: 0.01, left: 0.01, right: 0.01} } -// if (percentages == undefined) { percentages = {}; } -// if (percentages.axes == undefined) { percentages.axes = { x:0.1, y:0.1 } } -// if (percentages.space == undefined) { percentages.space = { w: 0.8, h: 0.6 } } -// -// // SVG width and height -// var svgSpace = { -// w: space.w * percentages.space.w, -// h: space.h * percentages.space.h -// }, -// -// // Space after removing margins -// chartSpace = { -// w: svgSpace.w - (margins.left * space.w) - (margins.right * space.w), -// h: svgSpace.h - (margins.top * space.h) - (margins.bottom * space.h) -// }, -// -// // main dimension of x and y axies -// // e.g. defines how tall x axis is as length is determined by plotRect.w -// axesSpace = { -// x: chartSpace.h * percentages.axes.x, -// y: chartSpace.w * percentages.axes.y -// }, -// -// // space left for drawing the chart properly (e.g. bars, violins, etc) -// drawingSpace = { -// x: chartSpace.w - axesSpace.y, -// y: chartSpace.h - axesSpace.x -// }, -// -// -// yAxisRect = { -// x: axesSpace.y + (margins.left * space.w), -// y: (margins.top * space.h), -// w: axesSpace.y, -// h: drawingSpace.y -// }, -// -// plotRect = { -// x: axesSpace.y + (margins.left * space.w), -// y: (margins.top * space.h), -// w: drawingSpace.x, -// h: drawingSpace.y -// }, -// -// xAxisRect = { -// x: axesSpace.y + (margins.left * space.w), -// y: (margins.top * space.h + plotRect.h), -// w: drawingSpace.x, -// h: axesSpace.x -// } -// -// -// var container = safeSelect(selection, 'svg', namespace) -// .style('width', svgSpace.w+'px') -// .style('height', svgSpace.h+'px') -// -// var axes = safeSelect(container, 'g', hypenate(namespace, 'axes')) -// -// // .attr('transform', "translate("+plotRect.x+","+plotRect.y+")"), -// -// var plot = safeSelect(container, 'g', hypenate(namespace, 'plot')) -// .attr('transform', "translate("+plotRect.x+","+plotRect.y+")") -// -// var xAxis = safeSelect(axes, 'g', hypenate(namespace, 'x-axis')) -// .attr('transform', "translate("+xAxisRect.x+","+xAxisRect.y+")") -// -// var yAxis = safeSelect(axes, 'g', hypenate(namespace, 'y-axis')) -// .attr('transform', "translate("+yAxisRect.x+","+yAxisRect.y+")") -// -// return { -// svg: { -// selection: container, -// rect: svgSpace -// }, -// plot: { -// selection: plot, -// rect: plotRect -// }, -// xAxis: { -// selection: xAxis, -// rect: xAxisRect -// }, -// yAxis: { -// selection: yAxis, -// rect: yAxisRect -// } -// } -// -// // return [plot, xAxis, yAxis] -// } -// -// +function calculateWidthOfSpacer(data, freeSpace, objectWidth, numberOfObjects, baseSpacerSize, overflowQ) { + if (overflowQ) { + // var limitedNumberOfObjects = numberOfObjects > 6 ? 6 : numberOfObjects + // var spaceLeft = freeSpace - limitedNumberOfObjects * objectWidth + // return spaceLeft / (limitedNumberOfObjects - 1) + return freeSpace * baseSpacerSize + } + var spacersAtEachLevel = spacersNeededAtEachLevel(data); + var totalSpacerPercent = total(spacersAtEachLevel.map(function(e, i) {return e * 1 / (i+1)})); + var baseSpacerSize = (freeSpace - (objectWidth * numberOfObjects)) / totalSpacerPercent; + // console.log(freeSpace, objectWidth, numberOfObjects, totalSpacerPercent) + // console.log(totalSpacerPercent, baseSpacerSize, totalSpacerPercent * baseSpacerSize) + return isNaN(baseSpacerSize) ? 0 : baseSpacerSize +} +/** +* Calculates number of spacers needed to seperate elements at each level. +* @param {Array[]} array list data (can be nested). If nested will create more complex spacer size +* @param {number} [level=0] current level, used in recusrion +* @param {Array} [levelData=[]] how many spacers needed at a given level +* @returns {Array} levelData +* +* @example +* array = [[1,2], [3,4]] +* // returns [1, 2] +* as at level=0 the only spacer needed is between [1,2] and [3,4] +* and at level=1 the only two spacers needed is between 1 and 2 as well as +* 3 and 4 since the spacer between 2 and 3 is handled at level=0 +*/ +function spacersNeededAtEachLevel (array, level, levelData ) { + if ( level == undefined ) { level = 0; } else { level += 1; } + if ( levelData == undefined ) { levelData = []; } + if ( level >= levelData.length ) { levelData.push(array.length - 1); } + else { levelData[level] += array.length - 1; } + array.map(function(e, i) { if (Array.isArray(e)) { spacersNeededAtEachLevel(e, level, levelData); } } ); + return levelData +} +var math = /*#__PURE__*/Object.freeze({ + round: round, + tickRange: tickRange, + partitionRangeInto: partitionRangeInto, + euclideanDistance: euclideanDistance, + quartiles: quartiles, + calculateWidthOfObject: calculateWidthOfObject, + calculateWidthOfSpacer: calculateWidthOfSpacer, + spacersNeededAtEachLevel: spacersNeededAtEachLevel +}); +function resizeDebounce(f, wait) { + var resize = debounce(function(){f();},wait); + window.addEventListener('resize', resize); +} +function debounce(func, wait, immediate) { + var timeout; + return function() { + var context = this, args = arguments; + var later = function() { + timeout = null; + if (!immediate) func.apply(context, args); + }; + var callNow = immediate && !timeout; + clearTimeout(timeout); + timeout = setTimeout(later, wait); + if (callNow) func.apply(context, args); + }; +} function setupStandardChartContainers( selection, @@ -680,28 +574,132 @@ function setupStandardChartContainers( } } - +var misc = /*#__PURE__*/Object.freeze({ + resizeDebounce: resizeDebounce, + setupStandardChartContainers: setupStandardChartContainers +}); /** -* Adds a clip-path rect and binds it to container -* @param {d3.selection} container in which to add the clip-path and to which to bind the cliping path to -* @param {Object} rect the coordinates (x, y, width, height) of the clip-path -* @param {string} namespace -* @returns {d3.selection} of the clip-path rect +* Hypenates all strings together +* @param {string[]} arguments +* @returns {string} "arg1-arg2-...-argn" */ -function cpRect(container, rect, namespace) { - var defs = safeSelect(container, 'defs', hypenate(namespace, 'definitions')); - var cp = safeSelect(defs, 'clipPath', hypenate(namespace, 'clip-path')) - .attr('id', hypenate(namespace, 'clip-path')); +function hypenate(){ return Array.prototype.slice.call(arguments).join('-') } - var cpRect = safeSelect(cp, 'rect') - .attr('x', rect.x) - .attr('y', rect.y) - .attr('width', rect.width) - .attr('height', rect.height); - defs.raise(); - // set clipping path to container +/** +* Trys to reduce text to fit in specified area, made for tick labels as called by +* @see{@link axis} +* @param {d3.selection} t container for specific axis tick +* @param {string} text to be the label of the passed axis tick +* @param {boolean} orient of the axis, true is horizontal, false is vertical +* @param {number} tickLength is the length of the text +* @param {number} space is the amount of availble space for the text and the tick to fit in +* @param {boolean} overflowQ whether or not allowed to go over the alloted space +* @returns {none} +*/ +function truncateText(t, text, orient, tickLength, space, overflowQ) { + var rect = t.node().getBoundingClientRect(); + t.text(text); + while (Math.max(rect.width, rect.height) > space - tickLength) { + text = String(text); + text = text.slice(0, text.length - 1); + t.text(text + '...'); + rect = t.node().getBoundingClientRect(); + if (text.length == 0) break + } +} + + +function truncateString(string, space, font) { + var chars = space / font; + if (chars < string.length) { + return string.slice(0, Math.round(chars-5)) + '...' + } else { + return string + } +} + +var str = /*#__PURE__*/Object.freeze({ + hypenate: hypenate, + truncateText: truncateText, + truncateString: truncateString +}); + +/** +* Extracts x and y of translate from transform property +* @param {string} transform transform property of svg element +* @returns {number[]} x, y of translate(x, y) +*/ +function getTranslation(transform) { + // Create a dummy g for calculation purposes only. This will never + // be appended to the DOM and will be discarded once this function + // returns. + var g = document.createElementNS('http://www.w3.org/2000/svg', 'g'); + // Set the transform attribute to the provided string value. + transform = transform == undefined ? 'translate(0,0)' : transform; + g.setAttributeNS(null, 'transform', transform); + // consolidate the SVGTransformList containing all transformations + // to a single SVGTransform of type SVG_TRANSFORM_MATRIX and get + // its SVGMatrix. + var matrix = g.transform.baseVal.consolidate().matrix; + // As per definition values e and f are the ones for the translation. + return [matrix.e, matrix.f]; +} + +/** +* recursively ascends element.parentElement to find a svg tag +* @param {Element} element +* @returns {Element | undefined} +*/ +function getContainingSVG$1(element) { + var parent = element.parentElement; + var tag = parent.tagName.toLowerCase(); + if (tag === 'svg') { return parent; } + if (tag === 'html') { return undefined; } + return getContainingSVG$1(parent); +} + + +/** +* Trys to use d3.selection to get element, if it doesnt exist, makes one +* @param {d3.selection} sel selection in which to try and find object +* @param {string} tag tag of which to try and select +* @param {string} [cls=''] class of tag to try and grab +* @returns {d3.selection} of either append or selected tag.cls within sel +*/ +function safeSelect(sel, tag, cls) { + var clsStr = cls == undefined ? '' : '.'+cls; + var sSel = sel.select(tag+clsStr).empty() + ? sel.append(tag) + : sel.select(tag+clsStr); + return sSel + .classed(clsStr.replace('.', ''), true) + .attr('transform', sSel.attr('transform') == undefined ? 'translate(0,0)' : sSel.attr('transform')) +} + + + +/** +* Adds a clip-path rect and binds it to container +* @param {d3.selection} container in which to add the clip-path and to which to bind the cliping path to +* @param {Object} rect the coordinates (x, y, width, height) of the clip-path +* @param {string} namespace +* @returns {d3.selection} of the clip-path rect +*/ +function cpRect(container, rect, namespace) { + var defs = safeSelect(container, 'defs', hypenate(namespace, 'definitions')); + var cp = safeSelect(defs, 'clipPath', hypenate(namespace, 'clip-path')) + .attr('id', hypenate(namespace, 'clip-path')); + + var cpRect = safeSelect(cp, 'rect') + .attr('x', rect.x) + .attr('y', rect.y) + .attr('width', rect.width) + .attr('height', rect.height); + + defs.raise(); + // set clipping path to container container.attr('clip-path', 'url(#'+ hypenate(namespace, 'clip-path')+')'); return cpRect @@ -748,96 +746,130 @@ function setupContainer(selection, namespace, rect, fill) { return objectContainer } +var sel = /*#__PURE__*/Object.freeze({ + getTranslation: getTranslation, + getContainingSVG: getContainingSVG$1, + safeSelect: safeSelect, + cpRect: cpRect, + bgRect: bgRect, + setupContainer: setupContainer +}); /** -* determines the width of an object for the calling plotting function -* @param {number} freeSpace how much space is avalible -* @param {number} numberOfObjects how many object do we need -* @param {number} minObjectWidth how small are these objects allowed to be -* @param {number} maxObjectWidth how large are these object allowed to be -* @param {number} sizeOfSpacer percent of freeSpace that a single spacer should take up (need numberOfObjects - 1 spacers) -* @param {boolean} overflowQ can we go beyond alloted space -* @returns {number} how large object should be -* function tries to keep object within min / max width, but wil default to -* 5e-10 (smallest consistenly visible by svg size of element) if overflowQ is false -*/ -function calculateWidthOfObject(freeSpace, numberOfObjects, minObjectWidth, maxObjectWidth, sizeOfSpacer, overflowQ) { - var sizeOfSpacer = - sizeOfSpacer == 0 || sizeOfSpacer > 1 - ? sizeOfSpacer - : freeSpace * sizeOfSpacer; - - var numberOfSpacers = numberOfObjects - 1; - var spaceTakenBySpacers = numberOfSpacers * sizeOfSpacer; - var remainingSpace = freeSpace - spaceTakenBySpacers; - remainingSpace = remainingSpace < 0 ? 0 : remainingSpace; - var objectWidth = remainingSpace / numberOfObjects; - - if ( overflowQ && minObjectWidth != undefined && objectWidth < minObjectWidth ) { objectWidth = minObjectWidth; } - // if ( maxObjectWidth != undefined && objectWidth > maxObjectWidth ) { objectWidth = maxObjectWidth } - if ( overflowQ && maxObjectWidth != undefined && objectWidth < maxObjectWidth ) { objectWidth = maxObjectWidth; } - return Math.max(objectWidth, 5e-10) + * calls console.group if d3sm.debugQ == true + * @param {string} name of the group + * @returns {undefined} + */ +function consoleGroup(name) { + if (window.d3sm.debugQ === true){ + console.group(name); + } } /** -* @param {Array[]} data list data (can be nested). If nested will create more complex spacer size -* @param {number} freeSpace how much space is avalible -* @param {number} objectWidth @see{@link calculateWidthOfObject} -* @param {number} numberOfObjects how many object do we need -* @param {number} baseSpacerSize percent of freeSpace that a single spacer should take up (need numberOfObjects - 1 spacers) -* @param {boolean} overflowQ can we go beyond alloted space -* @returns {number} returns size that spacer should be at level=0 -*/ -function calculateWidthOfSpacer(data, freeSpace, objectWidth, numberOfObjects, baseSpacerSize, overflowQ) { - if (overflowQ) { - // var limitedNumberOfObjects = numberOfObjects > 6 ? 6 : numberOfObjects - // var spaceLeft = freeSpace - limitedNumberOfObjects * objectWidth - // return spaceLeft / (limitedNumberOfObjects - 1) - return freeSpace * baseSpacerSize + * calls console.groupEnd if d3sm.debugQ == true + * @returns {undefined} + */ +function consoleGroupEnd() { + if (window.d3sm.debugQ === true){ + console.groupEnd(); } - var spacersAtEachLevel = spacersNeededAtEachLevel(data); - var totalSpacerPercent = total(spacersAtEachLevel.map(function(e, i) {return e * 1 / (i+1)})); - var baseSpacerSize = (freeSpace - (objectWidth * numberOfObjects)) / totalSpacerPercent; - // console.log(freeSpace, objectWidth, numberOfObjects, totalSpacerPercent) - // console.log(totalSpacerPercent, baseSpacerSize, totalSpacerPercent * baseSpacerSize) - return isNaN(baseSpacerSize) ? 0 : baseSpacerSize } - /** -* Calculates number of spacers needed to seperate elements at each level. -* @param {Array[]} array list data (can be nested). If nested will create more complex spacer size -* @param {number} [level=0] current level, used in recusrion -* @param {Array} [levelData=[]] how many spacers needed at a given level -* @returns {Array} levelData -* -* @example -* array = [[1,2], [3,4]] -* // returns [1, 2] -* as at level=0 the only spacer needed is between [1,2] and [3,4] -* and at level=1 the only two spacers needed is between 1 and 2 as well as -* 3 and 4 since the spacer between 2 and 3 is handled at level=0 -*/ -function spacersNeededAtEachLevel (array, level, levelData ) { - if ( level == undefined ) { level = 0; } else { level += 1; } - if ( levelData == undefined ) { levelData = []; } - if ( level >= levelData.length ) { levelData.push(array.length - 1); } - else { levelData[level] += array.length - 1; } - array.map(function(e, i) { if (Array.isArray(e)) { spacersNeededAtEachLevel(e, level, levelData); } } ); - return levelData + * Calls console.log if d3sm.debugQ == true + * @param {string} func name of the function logging + * @param {string} msg to log + * @param {Object} data to be logged along side the message + * @returns {undefined} + */ +function log(func, msg, data) { + if (window.d3sm.debugQ === true){ + console.log( + `%c[d3sm::${func}]:\t${msg}`, + [ + 'background: #6cd1ef', + 'border-radius: 5000px', + 'padding: 0px 2px', + 'font-size: 14px' + ].join(';') + ); + console.table(data); + // console.trace() + } } - +/** + * Calls console.warn if d3sm.debugQ == true + * @param {string} func name of the function warning + * @param {string} msg to display + * @param {Object} data to be displayed along side the message + * @returns {undefined} + */ +function warn(func, msg, data) { + if (window.d3sm.debugQ === true) + console.warn( + `%c[d3sm::${func}]:\t${msg}`, + [ + 'background: #ffd53e', + 'border-radius: 5000px', + 'padding: 0px 2px', + 'font-size: 14px' + ].join(';') + ); + console.table(data); +} +/** + * Calls the console.info if d3sm.debugQ == true + * @param {string} func name of the function providing info + * @param {string} msg to display + * @param {Object} data to be displayed along side the message + * @returns {undefined} + */ +function info(func, msg, data) { + if (window.d3sm.debugQ) + console.info( + `%c[d3sm::${func}]:\t${msg}`, + [ + 'background: #009ccd', + 'border-radius: 5000px', + 'padding: 0px 2px', + 'font-size: 14px' + ].join(';') + ); + console.table(data); +} /** -* Draws a whisker for @see{@link boxwhisker} -* @param {boolean} dir direction to draw whisker, should be either true (up, top) or false (down or bottom) -* @param {number} x starting x coordinate in which to draw whisker -* @param {number} y starting y coordinate in which to draw whisker -* @param {number} w width of space in which to draw whisker -* @param {number} h height of space in which to draw whisker -* @param {number} per percentage of w or h (depends on o) to make whisker + * Calls console.error if d3sm.debugQ == true + * @param {string} func name of the function which sends the error + * @param {string} msg to display + * @param {Object} data to be displayed along side the message + * @returns {undefined} + */ +function error(func, msg, data) { + if (window.d3sm.debugQ) + console.error(`[d3sm::${func}]:\t${msg}\t%o`,data); +} + +var con = /*#__PURE__*/Object.freeze({ + consoleGroup: consoleGroup, + consoleGroupEnd: consoleGroupEnd, + log: log, + warn: warn, + info: info, + error: error +}); + +/** +* Draws a whisker for @see{@link boxwhisker} +* @param {boolean} dir direction to draw whisker, should be either true (up, top) or false (down or bottom) +* @param {number} x starting x coordinate in which to draw whisker +* @param {number} y starting y coordinate in which to draw whisker +* @param {number} w width of space in which to draw whisker +* @param {number} h height of space in which to draw whisker +* @param {number} per percentage of w or h (depends on o) to make whisker * @param {boolean} o orientation, true is horizontal and false is vertical * @returns {string} representing the svg path (i.e. the d attribute for a path tag) */ @@ -870,39 +902,13 @@ function whiskerPath(dir, x, y, w, h, per, o) { return p } +var paths = /*#__PURE__*/Object.freeze({ + whiskerPath: whiskerPath +}); - - - - - - - - - - - -function resizeDebounce(f, wait) { - var resize = debounce(function(){f();},wait); - window.addEventListener('resize', resize); -} - - - -function debounce(func, wait, immediate) { - var timeout; - return function() { - var context = this, args = arguments; - var later = function() { - timeout = null; - if (!immediate) func.apply(context, args); - }; - var callNow = immediate && !timeout; - clearTimeout(timeout); - timeout = setTimeout(later, wait); - if (callNow) func.apply(context, args); - }; -} +let utils$1 = { + arr, color, math, misc, sel, str, con, paths +}; /******************************************************************************* ** ** @@ -1030,7 +1036,7 @@ function groupingSpacer() { x = horizontalQ ? window.outerWidth : 0, y = !horizontalQ ? window.outerWidth : 0, t = 'translate('+x+','+y+')'; - // if(y == undefined) {console.log(cur.node(), y, d)} + // if(y == undefined) {console.con.log(cur.node(), y, d)} return t }); }, @@ -1063,7 +1069,7 @@ function groupingSpacer() { x = horizontalQ ? window.outerWidth : 0, y = !horizontalQ ? window.outerWidth : 0, t = 'translate('+x+','+y+')'; - // if(y == undefined) {console.log(cur.node(), y, d)} + // if(y == undefined) {console.con.log(cur.node(), y, d)} return t }).remove(); }; @@ -2010,7 +2016,7 @@ function axis ( selection ) { function axis () { // for convenience in handling orientation specific values - var horizontalQ = hasQ(['top', 'bottom', 'horizontal'], orient) ? true : false; + var horizontalQ = utils$1.arr.hasQ(['top', 'bottom', 'horizontal'], orient) ? true : false; var verticalQ = !horizontalQ; // background cliping rectangle @@ -2041,7 +2047,7 @@ function axis ( selection ) { } - var container = setupContainer( selection, namespace, bgcpRect, backgroundFill ); + var container = utils$1.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); // defaults for text-anchor and text rotation if (orient == 'top') { @@ -2079,12 +2085,12 @@ function axis ( selection ) { : (grouping == undefined) ? (numberOfTicks != undefined) // ? (tickValues.length < numberOfTicks) - ? (tickRange(...d3.extent(tickValues), numberOfTicks)) + ? (utils$1.math.tickRange(...d3.extent(tickValues), numberOfTicks)) : tickValues : grouping; - var flatTickData = flatten(tickData); + var flatTickData = utils$1.arr.flatten(tickData); var numberOfObjects = flatTickData.length; var space = horizontalQ ? spaceX : spaceY; var extent = d3.extent(flatTickData); @@ -2108,16 +2114,16 @@ function axis ( selection ) { // calculate object size if needed objectSize = (objectSize == undefined) - ? calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ) + ? utils$1.math.calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ) : objectSize; // calculate spacer size if needed spacerSize = (spacerSize == undefined) - ? calculateWidthOfSpacer(flatTickData, space, objectSize, numberOfObjects, objectSpacer, overflowQ) + ? utils$1.math.calculateWidthOfSpacer(flatTickData, space, objectSize, numberOfObjects, objectSpacer, overflowQ) : spacerSize; - var objClass = hypenate(namespace, categoricalQ ? objectClass+'-categorical' : objectClass); + var objClass = utils$1.str.hypenate(namespace, categoricalQ ? objectClass+'-categorical' : objectClass); var spacerFunction = groupingSpacer() .horizontalQ(horizontalQ).scale(scale).moveby((categoricalQ?'category':'scale')).numberOfObjects(numberOfObjects) @@ -2125,12 +2131,12 @@ function axis ( selection ) { .transitionDuration(transitionDuration).easeFunc(easeFunc) .namespace(namespace); - var tickEnterAnimation = function(sel){ - var mt = scale(sel.datum()), + var tickEnterAnimation = function(sel$$1){ + var mt = scale(sel$$1.datum()), dist = scale(extent[1]) * 2, k = (mt < extent[1] / 2) ? 1 : -1; k = horizontalQ ? k * -1 : k; - sel.attr('transform', function (d, i) { + sel$$1.attr('transform', function (d, i) { var x = horizontalQ ? dist * k : 0, y = !horizontalQ ? dist * k : 0, @@ -2138,12 +2144,12 @@ function axis ( selection ) { return t }); }; - var tickExitAnimation = function(sel) { - var mt = scale(sel.datum()), + var tickExitAnimation = function(sel$$1) { + var mt = scale(sel$$1.datum()), dist = scale(extent[1]) * 2, k = (mt < extent[1] / 2) ? 1 : -1; k = horizontalQ ? k * -1 : k; - sel.transition().duration(transitionDuration).ease(easeFunc) + sel$$1.transition().duration(transitionDuration).ease(easeFunc) .style('opacity', 0) .attr('transform', function (d, i) { var @@ -2182,11 +2188,11 @@ function axis ( selection ) { - var labelNameGroup = safeSelect(selection, 'g', hypenate(namespace,'axis-name')); + var labelNameGroup = utils$1.sel.safeSelect(selection, 'g', utils$1.str.hypenate(namespace,'axis-name')); - var labelElement = safeSelect(labelNameGroup, 'text', hypenate(namespace, 'name')); + var labelElement = utils$1.sel.safeSelect(labelNameGroup, 'text', utils$1.str.hypenate(namespace, 'name')); if (labelElement != undefined) { labelElement.text(label); @@ -2232,8 +2238,8 @@ function axis ( selection ) { to use clip path to make things fit in fixed size. Have yet got this to work nicely. */ // var defs = d3.select(container.node().parentNode).select('defs') - // var tickLabelClipPath = safeSelect(defs, 'clipPath', hypenate(namespace,'tick-label-clip-path')).attr('id', hypenate(namespace,'tick-label-clip-path')) - // var tickLabelClipPathRect = safeSelect(tickLabelClipPath, 'rect', hypenate(namespace,'tick-label-clip-path-rect')) + // var tickLabelClipPath = utils.sel.safeSelect(defs, 'clipPath', utils.str.hypenate(namespace,'tick-label-clip-path')).attr('id', utils.str.hypenate(namespace,'tick-label-clip-path')) + // var tickLabelClipPathRect = utils.sel.safeSelect(tickLabelClipPath, 'rect', utils.str.hypenate(namespace,'tick-label-clip-path-rect')) // .attr('x', 0) // .attr('y', 0) // .attr('width', function(d, i){ @@ -2251,7 +2257,7 @@ function axis ( selection ) { var that = d3.select(this).style('opacity', 1); // make and move tick - var tick = safeSelect(that, 'line', hypenate(namespace,'tick')) + var tick = utils$1.sel.safeSelect(that, 'line', utils$1.str.hypenate(namespace,'tick')) .attr("x1", 0) .attr("x2", horizontalQ ? 0 : orient == "left" ? -tickLength : tickLength) .attr("y1", 0) @@ -2267,15 +2273,15 @@ function axis ( selection ) { }); // make and move label - var label = safeSelect(that, 'text', hypenate(namespace,'label')) + var label = utils$1.sel.safeSelect(that, 'text', utils$1.str.hypenate(namespace,'label')) .text(function(d, i){ - var s = typeof d == 'number' ? round(d, roundTo) : d; - s = truncateString(String(s), (horizontalQ ? spaceY : spaceX) - tickLength-tickLabelMargin-tickTickLabelSpacer, tickLabelFontSize * 0.45); + var s = typeof d == 'number' ? utils$1.math.round(d, roundTo) : d; + s = utils$1.str.truncateString(String(s), (horizontalQ ? spaceY : spaceX) - tickLength-tickLabelMargin-tickTickLabelSpacer, tickLabelFontSize * 0.45); return s }) .attr('font-size', tickLabelFontSize) .attr('text-anchor', tickLabelTextAnchor); - // truncateText(label, label.text(), orient, tickLength, horizontalQ ? spaceY : spaceX, overflowQ) + // utils.str.truncateText(label, label.text(), orient, tickLength, horizontalQ ? spaceY : spaceX, overflowQ) label.attr('transform', function(d, i) { var @@ -2318,11 +2324,11 @@ function axis ( selection ) { .on('mousemove', labelHover) .on('mouseout', labelHoverOff) .on('click', tickLabelOnClick); - // .attr('clip-path', 'url(#'+hypenate(namespace,'tick-label-clip-path')+')') + // .attr('clip-path', 'url(#'+utils.str.hypenate(namespace,'tick-label-clip-path')+')') // add guidlines as needed if (guideLinesQ) { - var gline = safeSelect(that, 'line', hypenate(namespace, 'guideline')) + var gline = utils$1.sel.safeSelect(that, 'line', utils$1.str.hypenate(namespace, 'guideline')) .transition().duration(transitionDuration).ease(easeFunc) .attr("x1", 0) .attr("x2", horizontalQ ? 0 : orient == "left" ? guidelineSpace : -guidelineSpace) @@ -2335,15 +2341,15 @@ function axis ( selection ) { t = 'translate('+x+','+y+')'; return t }); - } else { that.select('line.'+hypenate(namespace, 'guideline')).remove(); } + } else { that.select('line.'+utils$1.str.hypenate(namespace, 'guideline')).remove(); } }); // apply alternating guidline thickness if (guideLinesQ) { - container.selectAll('.'+hypenate(namespace,'guideline')) + container.selectAll('.'+utils$1.str.hypenate(namespace,'guideline')) .attr('stroke', function(d, i){ - if (i % 2 == 0) { return modifyHexidecimalColorLuminance(guideLineStroke, 0.8) } + if (i % 2 == 0) { return utils$1.color.modifyHexidecimalColorLuminance(guideLineStroke, 0.8) } return guideLineStroke }) .attr('stroke-width', function(d, i){ @@ -2357,7 +2363,7 @@ function axis ( selection ) { /*************************************************************************** ** Make the line of the axis ***************************************************************************/ - var line = safeSelect(selection, 'path', hypenate(namespace,'line')) + var line = utils$1.sel.safeSelect(selection, 'path', utils$1.str.hypenate(namespace,'line')) // .attr('x1', 0) // .attr('x2', horizontalQ ? spaceX : 0) // .attr('y1', 0) @@ -2377,21 +2383,21 @@ function axis ( selection ) { // hover of label show full text label in case it is truncated function labelHover(d, i){ var t = d3.select(this).style('fill', 'red'); - d3.select(t.node().parentNode).select("line."+hypenate(namespace,'tick')) + d3.select(t.node().parentNode).select("line."+utils$1.str.hypenate(namespace,'tick')) .attr("stroke", 'red') .attr("stroke-width", tickStrokeWidth*2); if (guideLinesQ) { - d3.select(t.node().parentNode).select('line.'+hypenate(namespace, 'guideline')) + d3.select(t.node().parentNode).select('line.'+utils$1.str.hypenate(namespace, 'guideline')) .attr('stroke', 'red') .attr('stroke-width', guideLineStrokeWidth*2); } - var s = typeof d == 'number' ? round(d, roundTo) : d; + var s = typeof d == 'number' ? utils$1.math.round(d, roundTo) : d; var m = d3.mouse(d3.select('html').node()); - var div = safeSelect(d3.select('body'), 'div', hypenate(namespace,'guideline-tooltip')) - .attr('id', hypenate(namespace,'guideline-tooltip')) + var div = utils$1.sel.safeSelect(d3.select('body'), 'div', utils$1.str.hypenate(namespace,'guideline-tooltip')) + .attr('id', utils$1.str.hypenate(namespace,'guideline-tooltip')) .style('position', 'absolute') .style('left', (d3.event.pageX+15)+'px') .style('top', (d3.event.pageY+15)+'px') @@ -2408,7 +2414,7 @@ function axis ( selection ) { .style('border-style', 'solid') .style('border-width', 2); - var text = safeSelect(div, 'div') + var text = utils$1.sel.safeSelect(div, 'div') .text(tickLabelOnHoverFunc(s, i)) .style('color', 'black') .style('align-self', 'center'); @@ -2421,15 +2427,15 @@ function axis ( selection ) { function labelHoverOff(d, i){ var t = d3.select(this).style('fill', 'black'); - d3.select(t.node().parentNode).select("line."+hypenate(namespace,'tick')) + d3.select(t.node().parentNode).select("line."+utils$1.str.hypenate(namespace,'tick')) .attr("stroke", tickStroke) .attr("stroke-width", tickStrokeWidth); if (guideLinesQ) { - var gline = d3.select(t.node().parentNode).select('line.'+hypenate(namespace, 'guideline')); + var gline = d3.select(t.node().parentNode).select('line.'+utils$1.str.hypenate(namespace, 'guideline')); var minorQ = gline.attr('minor'); gline.attr('stroke', function(d, ii){ - if (minorQ == 'true') { return modifyHexidecimalColorLuminance(guideLineStroke, 0.8) } + if (minorQ == 'true') { return utils$1.color.modifyHexidecimalColorLuminance(guideLineStroke, 0.8) } return guideLineStroke }) .attr('stroke-width', function(d, ii){ @@ -2437,7 +2443,7 @@ function axis ( selection ) { return guideLineStrokeWidth }); } - d3.select("#"+hypenate(namespace,'guideline-tooltip')).remove(); + d3.select("#"+utils$1.str.hypenate(namespace,'guideline-tooltip')).remove(); } @@ -2445,6 +2451,164 @@ function axis ( selection ) { return axis } +/******************************************************************************* +** ** +** ** +** TOOLTIP ** +** ** +** ** +*******************************************************************************/ +/** + * Produces a function for handling the tooltip + * + * {@link https://sumneuron.gitlab.io/d3sm/demos/tooltip-design/index.html Demo} + * @param {d3.selection} selection + * @returns {tooltip} + * @namespace tooltip + */ +function tooltip( selection ) { + + var + keys, + values, + header, + data, + selection; + + /** + * Gets / sets the keys to be displayed in the tooltip. + * If not set, uses d3.keys(data[key]) + * @param {string[]} [_=none] + * @returns {tooltip | string[]} + * @memberof tooltip + */ + tooltip.keys = function(_){return arguments.length ? (keys = _, tooltip) : keys}; + /** + * Gets / sets the values to be displayed next to the keys. + * If not set, uses data[key][keys[i]]. + * If a function, gets passed currentData (data[key]) and keys[i]. + * @param {*[]} [_=none] + * @returns {tooltip | *[]} + * @memberof tooltip + */ + tooltip.values = function(_){return arguments.length ? (values = _, tooltip) : values}; + /** + * Gets / sets the header to be displayed in the tooltip. + * If not set, uses key + * @param {string} [_=none] + * @returns {tooltip | string} + * @memberof tooltip + */ + tooltip.header = function(_){return arguments.length ? (header = _, tooltip) : header}; + /** + * Gets / sets the data (over the selection) to be used for the tooltip + * @param {Object} [_=none] + * @returns {tooltip | Object} + * @memberof tooltip + */ + tooltip.data = function(_){return arguments.length ? (data = _, tooltip) : data}; + /** + * Gets / sets the selection for the tooltip to be applied on + * @param {d3.selection} [_=none] + * @returns {tooltip | d3.selection} + * @memberof tooltip + */ + tooltip.selection = function(_){return arguments.length ? (selection = _, tooltip) : selection}; + + /** + * Bind, via selection.on(), the mousemove and mouseout events + * @returns undefined + */ + function tooltip( ) { + selection.on('mouseover', mousemove); + selection.on('mousemove', mousemove); + selection.on('mouseout', function(){ d3.selectAll(".d3sm-tooltip").remove();}); + } + + + /** + * Produces the tooltip on mousemove + * @param {string} key of the object targeted by the mousemove + * @param {number} i (index) of the object targeted by mousemove + * @memberof tooltip + * @private + */ + function mousemove(key, i) { + consoleGroup('d3sm-tooltip'); + var currentData = data[key]; + + var [x, y] = d3.mouse(d3.select("html").node()); + log('tooltip', 'mousemove detected',{key: key, index: i, x:x, y:y}); + log('tooltip', 'current data', currentData); + + + + var div = safeSelect(d3.select('html'), 'tooltip', 'd3sm-tooltip') + .classed('card', true) + .style('max-width', '300px') + .style('background-color', "#212529") + .style('color', 'white'); + + + + var cardBody = safeSelect(div, 'div', 'card-body'); + var cardTitle = safeSelect(cardBody, 'h5', 'card-title') + .text(header == undefined ? key : typeof header == 'function' ? header(key, currentData, i) : header) + .style('color', 'cyan'); + + + var table = safeSelect(cardBody, 'table', 'table').classed('table-dark', true); + var tBody = safeSelect(table, 'tbody'); + + tBody = tBody.selectAll('tr'); + tBody= tBody.data(keys == undefined ? d3.keys(currentData): keys); + tBody.exit().remove(); + + + var tr = tBody.enter().append('tr').style('max-width', '300px'); + tr.append('td').attr('class', function(d, i){return 'tooltip-key'}); + tr.append('td').attr('class', function(d, i, j){return 'tooltip-value'}) + .attr('tooltip-row-index', function(d, i){return i}); + + // tBody = tBody.merge(tr) + consoleGroup('tooltip-rows'); + tBody.selectAll('.tooltip-key').text(function(d, i){return d}); + tBody.selectAll('tr .tooltip-value') + .text(function(d, i){ + log('tooltip', 'trying to set value', {rowKey: d, rowIndex: i}); + var i = d3.select(this).attr('tooltip-row-index'); + var v = currentData[d]; + + + if (values != undefined) {v = values[i]; if(typeof v == "function") {v = v(currentData, d);}} + return typeof v == 'number' ? round(v, 5) : v + }); + consoleGroupEnd(); + consoleGroupEnd(); + + x += 15; + // x += 15 + var bbox = div.node().getBoundingClientRect(); + if (x + bbox.width > window.innerWidth - window.scrollX) { x = d3.event.pageX - bbox.width - 15; } + if (y + bbox.height > window.innerHeight - window.scrollY) { y = d3.event.pageY - bbox.height - 15; } + div.style('position') == "relative" + ? div.style('position', 'absolute').style('left', x+'px').style('top', y+'px') + : div.style('left', x+'px').style('top', y+'px'); + // .transition().duration(200).ease(d3.easeSin) + + // if (bbox.x + bbox.width > window.innerWidth) { + // div.style('left', (d3.event.pageX-15-bbox.width)+'px') + // } + // if (bbox.y + bbox.height > window.innerHeight) { + // div.style('top', (d3.event.pageY-15-bbox.height)+'px') + // } + + div.attr('z-index', 10000); + } + + return tooltip +} + /** * Creates a colorFunction * @constructor colorFunction @@ -2733,1857 +2897,1539 @@ function colorFunction() { /******************************************************************************* ** ** ** ** -** TOOLTIP ** +** SCATTER ** ** ** ** ** *******************************************************************************/ /** - * Produces a function for handling the tooltip + * Creates a scatter * - * {@link https://sumneuron.gitlab.io/d3sm/demos/tooltip-design/index.html Demo} + * {@link https://sumneuron.gitlab.io/d3sm/demos/scatter/index.html Demo} + * @constructor scatter * @param {d3.selection} selection - * @returns {tooltip} - * @namespace tooltip + * @namespace scatter + * @returns {function} scatter */ -function tooltip( selection ) { +function scatter ( selection ) { var - keys, - values, - header, + /** + * Data to plot. Assumed to be a object, where each key corresponds to a point + * (see {@link scatter#data}) + * @param {Object} [data=undefined] + * @memberof scatter# + * @property + */ data, - selection; - /** - * Gets / sets the keys to be displayed in the tooltip. - * If not set, uses d3.keys(data[key]) - * @param {string[]} [_=none] - * @returns {tooltip | string[]} - * @memberof tooltip - */ - tooltip.keys = function(_){return arguments.length ? (keys = _, tooltip) : keys}; + * Amount of horizontal space (in pixels) avaible to render the scatter in + * (see {@link scatter#spaceX}) + * @param {number} [spaceX=undefined] + * @memberof scatter# + * @property + */ + spaceX, /** - * Gets / sets the values to be displayed next to the keys. - * If not set, uses data[key][keys[i]]. - * If a function, gets passed currentData (data[key]) and keys[i]. - * @param {*[]} [_=none] - * @returns {tooltip | *[]} - * @memberof tooltip - */ - tooltip.values = function(_){return arguments.length ? (values = _, tooltip) : values}; - /** - * Gets / sets the header to be displayed in the tooltip. - * If not set, uses key - * @param {string} [_=none] - * @returns {tooltip | string} - * @memberof tooltip - */ - tooltip.header = function(_){return arguments.length ? (header = _, tooltip) : header}; - /** - * Gets / sets the data (over the selection) to be used for the tooltip - * @param {Object} [_=none] - * @returns {tooltip | Object} - * @memberof tooltip - */ - tooltip.data = function(_){return arguments.length ? (data = _, tooltip) : data}; - /** - * Gets / sets the selection for the tooltip to be applied on - * @param {d3.selection} [_=none] - * @returns {tooltip | d3.selection} - * @memberof tooltip - */ - tooltip.selection = function(_){return arguments.length ? (selection = _, tooltip) : selection}; - - /** - * Bind, via selection.on(), the mousemove and mouseout events - * @returns undefined - */ - function tooltip( ) { - selection.on('mouseover', mousemove); - selection.on('mousemove', mousemove); - selection.on('mouseout', function(){ d3.selectAll(".d3sm-tooltip").remove();}); - } - + * Amount of vertical space (in pixels) avaible to render the scatter in + * (see {@link scatter.spaceY}) + * @param {number} [spaceY=undefined] + * @memberof scatter# + * @property + */ + spaceY, /** - * Produces the tooltip on mousemove - * @param {string} key of the object targeted by the mousemove - * @param {number} i (index) of the object targeted by mousemove - * @memberof tooltip - * @private - */ - function mousemove(key, i) { - consoleGroup('d3sm-tooltip'); - var currentData = data[key]; - - var [x, y] = d3.mouse(d3.select("html").node()); - log('tooltip', 'mousemove detected',{key: key, index: i, x:x, y:y}); - log('tooltip', 'current data', currentData); - - - - var div = safeSelect(d3.select('html'), 'tooltip', 'd3sm-tooltip') - .classed('card', true) - .style('max-width', '300px') - .style('background-color', "#212529") - .style('color', 'white'); - - - - var cardBody = safeSelect(div, 'div', 'card-body'); - var cardTitle = safeSelect(cardBody, 'h5', 'card-title') - .text(header == undefined ? key : typeof header == 'function' ? header(key, currentData, i) : header) - .style('color', 'cyan'); - - - var table = safeSelect(cardBody, 'table', 'table').classed('table-dark', true); - var tBody = safeSelect(table, 'tbody'); - - tBody = tBody.selectAll('tr'); - tBody= tBody.data(keys == undefined ? d3.keys(currentData): keys); - tBody.exit().remove(); - - - var tr = tBody.enter().append('tr').style('max-width', '300px'); - tr.append('td').attr('class', function(d, i){return 'tooltip-key'}); - tr.append('td').attr('class', function(d, i, j){return 'tooltip-value'}) - .attr('tooltip-row-index', function(d, i){return i}); - - // tBody = tBody.merge(tr) - consoleGroup('tooltip-rows'); - tBody.selectAll('.tooltip-key').text(function(d, i){return d}); - tBody.selectAll('tr .tooltip-value') - .text(function(d, i){ - log('tooltip', 'trying to set value', {rowKey: d, rowIndex: i}); - var i = d3.select(this).attr('tooltip-row-index'); - var v = currentData[d]; - - - if (values != undefined) {v = values[i]; if(typeof v == "function") {v = v(currentData, d);}} - return typeof v == 'number' ? round(v, 5) : v - }); - consoleGroupEnd(); - consoleGroupEnd(); - - x += 15; - // x += 15 - var bbox = div.node().getBoundingClientRect(); - if (x + bbox.width > window.innerWidth - window.scrollX) { x = d3.event.pageX - bbox.width - 15; } - if (y + bbox.height > window.innerHeight - window.scrollY) { y = d3.event.pageY - bbox.height - 15; } - div.style('position') == "relative" - ? div.style('position', 'absolute').style('left', x+'px').style('top', y+'px') - : div.style('left', x+'px').style('top', y+'px'); - // .transition().duration(200).ease(d3.easeSin) - - // if (bbox.x + bbox.width > window.innerWidth) { - // div.style('left', (d3.event.pageX-15-bbox.width)+'px') - // } - // if (bbox.y + bbox.height > window.innerHeight) { - // div.style('top', (d3.event.pageY-15-bbox.height)+'px') - // } - - div.attr('z-index', 10000); - } - - return tooltip -} - -/******************************************************************************* -** ** -** ** -** BAR ** -** ** -** ** -*******************************************************************************/ - -/** - * Creates a bar - * - * {@link https://sumneuron.gitlab.io/d3sm/demos/bar-chart-same-data-complex-grouping/index.html Demo} - * @constructor bar - * @param {d3.selection} selection - * @namespace bar - * @returns {function} bar - */ -function bar ( selection ) { - /* - Assumes that data is list an object. - - The keys of data will be bound to the bars. - The valueExtractor function will extract the value from data[key]. - - Grouping can be used if desired. It should be an arbitrary complex list where - the values are strings matching keys in data. + * The scale for which scatter x values should be transformed by + * @param {d3.scale} [scaleX=d3.scaleLinear] + * @memberof scatter# + * @property */ - var + scaleX = d3.scaleLinear(), /** - * Data to plot. Assumed to be a object, where each key corresponds to a bar - * (see {@link bar#data}) - * @param {Object} [data=undefined] - * @memberof bar# + * The padding for the domain of the scaleX (see {@link scatter#scaleX}) + * @param {number} [domainPaddingX=0.5] + * @memberof scatter# * @property */ - data, + domainPaddingX = 0.5, /** - * Which direction to render the bars in - * (see {@link bar#orient}) - * @param {number} [orient='horizontal'] - * @memberof bar# + * The function for getting the x value of the current point + * @param {function} [valueExtractorX=function(d, i){return data[d]['x']}] + * @memberof scatter# * @property */ - orient='horizontal', + valueExtractorX = function(d, i) {return data[d]['x']}, + /** - * Amount of horizontal space (in pixels) avaible to render the bar in - * (see {@link bar#spaceX}) - * @param {number} [spaceX=undefined] - * @memberof bar# + * The scale for which scatter y values should be transformed by + * @param {d3.scale} [scaleY=d3.scaleLinear] + * @memberof scatter# * @property */ - spaceX, + scaleY = d3.scaleLinear(), /** - * Amount of vertical space (in pixels) avaible to render the bar in - * (see {@link bar.spaceY}) - * @param {number} [spaceY=undefined] - * @memberof bar# + * The padding for the domain of the scaleY (see {@link scatter#scaleY}) + * @param {number} [domainPaddingY=0.5] + * @memberof scatter# * @property */ - spaceY, - + domainPaddingY = 0.5, /** - * Whether or not to allow bar to render elements pass the main spatial dimension - * given the orientation (see {@link bar#orient}), where {@link bar#orient}="horizontal" - * the main dimension is {@link bar#spaceX} and where {@link bar#orient}="vertical" - * the main dimension is {@link bar#spaceY} - * @param {boolean} [overflowQ=false] - * @memberof bar# + * The function for getting the y value of the current point + * @param {function} [valueExtractorY=function(d, i){return data[d]['y']}] + * @memberof scatter# * @property */ - overflowQ = false, + valueExtractorY = function(d, i) {return data[d]['y']}, + /** - * An array - putatively of other arrays - depicting how bars should be arranged - * @param {Array[]} [grouping=undefined] - * @memberof bar# + * The scale for which scatter r values should be transformed by + * @param {d3.scale} [scaleR=d3.scaleLinear] + * @memberof scatter# * @property */ - grouping, - + scaleR = d3.scaleLinear(), /** - * How to get the value of the bar - * @param {function} [valueExtractor=function(key, index) { return data[key] }] - * @memberof bar# + * The padding for the domain of the scaleR (see {@link scatter#scaleR}) + * @param {number} [domainPaddingR=0.5] + * @memberof scatter# * @property */ - valueExtractor = function(key, index) { return data[key] }, + domainPaddingR = 0.5, /** - * How to sort the bars - if {@link bar#grouping} is not provided. - * @param {function} [sortingFunction=function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}] - * @memberof bar# + * The function for getting the r value of the current point + * @param {function} [valueExtractorR=function(d, i){return 2}] + * @memberof scatter# * @property */ - sortingFunction = function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}, - + valueExtractorR = function(d, i) {return 2}, /** - * The scale for which bar values should be transformed by - * @param {d3.scale} [scale=d3.scaleLinear] - * @memberof bar# + * The min radius a point can have + * @param {function} [minRadius=2] + * @memberof scatter# * @property */ - scale = d3.scaleLinear(), + minRadius = 2, /** - * The padding for the domain of the scale (see {@link bar#scale}) - * @param {number} [domainPadding=0.5] - * @memberof bar# + * The min radius a point can have + * @param {function} [maxRadius=10] + * @memberof scatter# * @property */ - domainPadding = 0.5, + maxRadius = 10, /** - * Default space for the spacer (percentage) of main dimension given the orientation - * (see {@link bar#orient}), where {@link bar#orient}="horizontal" - * the main dimension is {@link bar#spaceX} and where {@link bar#orient}="vertical" - * the main dimension is {@link bar#spaceY} between bars - * @param {number} [objectSpacer=0.05] - * @memberof bar# - * @property - */ - objectSpacer = 0.05, - /** - * The minimum size that an object can be - * @param {number} [minObjectSize=50] - * @memberof bar# - * @property - */ - minObjectSize = 50, - /** - * The maximum size that an object can be - * @param {number} [maxObjectSize=100] - * @memberof bar# - * @property - */ - maxObjectSize = 100, - - /** - * The stroke width of the bars - * @param {number} [barStrokeWidth=2] - * @memberof bar# + * The stroke width of the points + * @param {number} [pointStrokeWidth=2] + * @memberof scatter# * @property */ - barStrokeWidth = 2, + pointStrokeWidth = 2, /** * Instance of ColorFunction * @param {function} [colorFunction = colorFunction()] - * @memberof bar# + * @memberof scatter# * @property */ colorFunction$$1 = colorFunction(), - - /** * Color of the background * @param {string} [backgroundFill="transparent"] - * @memberof bar# + * @memberof scatter# * @property */ backgroundFill = 'transparent', /** - * Namespace for all items made by this instance of bar - * @param {string} [namespace="d3sm-bar"] - * @memberof bar# + * Namespace for all items made by this instance of scatter + * @param {string} [namespace="d3sm-scatter"] + * @memberof scatter# * @property */ - namespace = 'd3sm-bar', + namespace = 'd3sm-scatter', /** - * Class name for bar container ( element) - * @param {string} [objectClass="bar"] - * @memberof bar# + * Class name for scatter container ( element) + * @param {string} [objectClass="scatter-point"] + * @memberof scatter# * @property */ - objectClass = 'bar', - + objectClass = 'scatter-point', /** * Duration of all transitions of this element * @param {number} [transitionDuration=1000] - * @memberof bar# + * @memberof scatter# * @property */ transitionDuration = 1000, /** * Easing function for transitions * @param {d3.ease} [easeFunc=d3.easeExp] - * @memberof bar# + * @memberof scatter# * @property */ easeFunc = d3.easeExp, // useful values to extract to prevent re-calculation /** - * The keys of the bars - * @param {string[]} [barKeys=undefined] - * @memberof bar# + * The keys of the points + * @param {string[]} [pointKeys=undefined] + * @memberof scatter# * @property */ - barKeys, + pointKeys, /** - * The values of the bars - * @param {number[]} [barValues=undefined] - * @memberof bar# + * The x values of the points + * @param {number[]} [valuesX=undefined] + * @memberof scatter# * @property */ - barValues, + valuesX, /** - * The objectSize (actual width) used by the bars - * @param {number} [objectSize=undefined] - * @memberof bar# + * The y values of the points + * @param {number[]} [valuesY=undefined] + * @memberof scatter# * @property */ - objectSize, + valuesY, /** - * The spacerSize (actual width) used by the spacers between the bars - * @param {number} [spacerSize=undefined] - * @memberof bar# + * The r values of the points + * @param {number[]} [valuesR=undefined] + * @memberof scatter# * @property */ - spacerSize, + valuesR, + /** * Instance of Tooltip * @param {function} [tooltip=tooltip()] - * @memberof bar# + * @memberof scatter# * @property */ - tooltip$$1 = tooltip(), - barPercent = 1; + tooltip$$1 = tooltip(); /** * Gets or sets the selection in which items are manipulated * @param {d3.selection} [_=none] - * @returns {bar | d3.selection} - * @memberof bar + * @returns {scatter | d3.selection} + * @memberof scatter * @property * by default selection = selection */ - bar.selection = function(_) { return arguments.length ? (selection = _, bar) : selection; }; + scatter.selection = function(_) { return arguments.length ? (selection =_, scatter) : selection}; /** * Gets or sets the data - * (see {@link bar#data}) - * @param {number} [_=none] - * @returns {bar | object} - * @memberof bar - * @property - */ - bar.data = function(_) { return arguments.length ? (data = _, bar) : data; }; - /** - * Gets or sets the orient of the bars - * (see {@link bar#orient}) + * (see {@link scatter#data}) * @param {number} [_=none] - * @returns {bar | object} - * @memberof bar + * @returns {scatter | object} + * @memberof scatter * @property */ - bar.orient = function(_) { return arguments.length ? (orient = _, bar) : orient; }; + scatter.data = function(_) { return arguments.length ? (data =_, scatter) : data}; /** * Gets or sets the amount of horizontal space in which items are manipulated - * (see {@link bar#spaceX}) + * (see {@link scatter#spaceX}) * @param {number} [_=none] should be a number > 0 - * @returns {bar | number} - * @memberof bar + * @returns {scatter | number} + * @memberof scatter * @property * by default spaceX = undefined */ - bar.spaceX = function(_) { return arguments.length ? (spaceX = _, bar) : spaceX; }; + scatter.spaceX = function(_) { return arguments.length ? (spaceX =_, scatter) : spaceX}; /** * Gets or sets the amount of vertical space in which items are manipulated - * (see {@link bar#spaceY}) + * (see {@link scatter#spaceY}) * @param {number} [_=none] should be a number > 0 - * @returns {bar | number} - * @memberof bar + * @returns {scatter | number} + * @memberof scatter * @property * by default spaceY = undefined */ - bar.spaceY = function(_) { return arguments.length ? (spaceY = _, bar) : spaceY; }; + scatter.spaceY = function(_) { return arguments.length ? (spaceY =_, scatter) : spaceY}; + + /** - * Gets / sets whether or not bar is allowed to go beyond specified dimensions - * (see {@link bar#spaceX}) - * @param {boolean} [_=none] - * @returns {bar | boolean} - * @memberof bar - * @property - * by default overflowQ = false - */ - bar.overflowQ = function(_) { return arguments.length ? (overflowQ = _, bar) : overflowQ; }; - /** - * Gets / sets the grouping of the bars - * (see {@link bar#grouping}) - * @param {Array[]} [_=none] - * @returns {bar | Array[]} - * @memberof bar + * Gets / sets the x scale for which the scatter x values should be transformed by + * (see {@link scatter#scaleX}) + * @param {d3.scale} [_=none] + * @returns {scatter | d3.scale} + * @memberof scatter * @property - * by default grouping = undefined + * by default scaleX = d3.scaleLinear() */ - bar.grouping = function(_) { return arguments.length ? (grouping = _, bar) : grouping; }; + scatter.scaleX = function(_) { return arguments.length ? (scaleX =_, scatter) : scaleX}; /** - * Gets / sets the valueExtractor - * (see {@link bar#valueExtractor}) - * @param {function} [_=none] - * @returns {bar | function} - * @memberof bar + * Gets / sets the padding for the domain of the x scale + * (see {@link scatter#domainPaddingX}) + * @param {number} [_=none] + * @returns {scatter | number} + * @memberof scatter * @property - * by default valueExtractor = function(key, index) { return data[key] }, + * by default domainPaddingX = 0.5 */ - bar.valueExtractor = function(_) { return arguments.length ? (valueExtractor = _, bar) : valueExtractor; }; + scatter.domainPaddingX = function(_) { return arguments.length ? (domainPaddingX =_, scatter) : domainPaddingX}; /** - * Gets / sets the sortingFunction - * (see {@link bar#sortingFunction}) + * Gets / sets the valueExtractorX + * (see {@link scatter#valueExtractorX}) * @param {function} [_=none] - * @returns {bar | function} - * @memberof bar + * @returns {scatter | function} + * @memberof scatter * @property - * by default sortingFunction = function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}, + * by default valueExtractorX = function(key, index) { return data[key]['x'] } */ - bar.sortingFunction = function(_) { return arguments.length ? (sortingFunction = _, bar) : sortingFunction; }; + scatter.valueExtractorX = function(_) { return arguments.length ? (valueExtractorX =_, scatter) : valueExtractorX}; + + /** - * Gets / sets the scale for which the bar values should be transformed by - * (see {@link bar#scale}) + * Gets / sets the y scale for which the scatter y values should be transformed by + * (see {@link scatter#scaleY}) * @param {d3.scale} [_=none] - * @returns {bar | d3.scale} - * @memberof bar + * @returns {scatter | d3.scale} + * @memberof scatter * @property - * by default scale = d3.scaleLinear() + * by default scaleY = d3.scaleLinear() */ - bar.scale = function(_) { return arguments.length ? (scale = _, bar) : scale; }; + scatter.scaleY = function(_) { return arguments.length ? (scaleY =_, scatter) : scaleY}; /** - * Gets / sets the padding for the domain of the scale - * (see {@link bar#domainPadding}) + * Gets / sets the padding for the domain of the y scale + * (see {@link scatter#domainPaddingY}) * @param {number} [_=none] - * @returns {bar | number} - * @memberof bar + * @returns {scatter | number} + * @memberof scatter * @property - * by default domainPadding = 0.5 + * by default domainPaddingY = 0.5 */ - bar.domainPadding = function(_) { return arguments.length ? (domainPadding = _, bar) : domainPadding; }; + scatter.domainPaddingY = function(_) { return arguments.length ? (domainPaddingY =_, scatter) : domainPaddingY}; /** - * Gets / sets objectSpacer - * (see {@link bar#objectSpacer}) - * @param {number} [_=none] - * @returns {bar | number} - * @memberof bar + * Gets / sets the valueExtractorY + * (see {@link scatter#valueExtractorY}) + * @param {function} [_=none] + * @returns {scatter | function} + * @memberof scatter * @property - * by default objectSpacer = 0.05 + * by default valueExtractorY = function(key, index) { return data[key]['y'] } */ - bar.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, bar) : objectSpacer; }; + scatter.valueExtractorY = function(_) { return arguments.length ? (valueExtractorY =_, scatter) : valueExtractorY}; + + /** - * Gets / sets the minObjectSize - * (see {@link bar#minObjectSize}) - * @param {number} [_=none] - * @returns {bar | number} - * @memberof bar + * Gets / sets the r scale for which the scatter r values should be transformed by + * (see {@link scatter#scaleR}) + * @param {d3.scale} [_=none] + * @returns {scatter | d3.scale} + * @memberof scatter * @property - * by default minObjectSize = 50 + * by default scaleR = d3.scaleLinear() */ - bar.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, bar) : minObjectSize; }; + scatter.scaleR = function(_) { return arguments.length ? (scaleR =_, scatter) : scaleR}; /** - * Gets / sets the maxObjectSize - * (see {@link bar#maxObjectSize}) + * Gets / sets the padding for the domain of the r scale + * (see {@link scatter#domainPaddingR}) * @param {number} [_=none] - * @returns {bar | number} - * @memberof bar + * @returns {scatter | number} + * @memberof scatter * @property - * by default maxObjectSize = 100 + * by default domainPaddingR = 0.5 */ - bar.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, bar) : maxObjectSize; }; - + scatter.domainPaddingR = function(_) { return arguments.length ? (domainPaddingR =_, scatter) : domainPaddingR}; /** - * Gets / sets the barStrokeWidth - * (see {@link bar#barStrokeWidth}) - * @param {number} [_=none] - * @returns {bar | number} - * @memberof bar - * @property - * by default barStrokeWidth = 2 + * Gets / sets the valueExtractorR + * (see {@link scatter#valueExtractorR}) + * @param {function} [_=none] + * @returns {scatter | function} + * @memberof scatter + * @property + * by default valueExtractorR = function(key, index) { return data[key]['r'] } */ - bar.barStrokeWidth = function(_) { return arguments.length ? (barStrokeWidth = _, bar) : barStrokeWidth; }; + scatter.valueExtractorR = function(_) { return arguments.length ? (valueExtractorR =_, scatter) : valueExtractorR}; + /** + * Gets / sets the minRadius + * (see {@link scatter#minRadius}) + * @param {number} [_=none] + * @returns {scatter | number} + * @memberof scatter + * @property + * by default minRadius = 2 + */ + scatter.minRadius = function(_) { return arguments.length ? (minRadius =_, scatter) : minRadius}; + /** + * Gets / sets the maxRadius + * (see {@link scatter#maxRadius}) + * @param {number} [_=none] + * @returns {scatter | number} + * @memberof scatter + * @property + * by default maxRadius = 10 + */ + scatter.maxRadius = function(_) { return arguments.length ? (maxRadius =_, scatter) : maxRadius}; + + /** + * Gets / sets the pointStrokeWidth + * (see {@link scatter#pointStrokeWidth}) + * @param {number} [_=none] + * @returns {scatter | number} + * @memberof scatter + * @property + * by default pointStrokeWidth = 2 + */ + scatter.pointStrokeWidth = function(_) { return arguments.length ? (pointStrokeWidth =_, scatter) : pointStrokeWidth}; /** * Gets / sets the colorFunction - * (see {@link bar#colorFunction}) + * (see {@link scatter#colorFunction}) * @param {number} [_=none] - * @returns {bar | number} - * @memberof bar + * @returns {scatter | number} + * @memberof scatter * @property * by default colorFunction = colorFunction() */ - bar.colorFunction = function(_) { return arguments.length ? (colorFunction$$1 = _, bar) : colorFunction$$1; }; - + scatter.colorFunction = function(_) { return arguments.length ? (colorFunction$$1 =_, scatter) : colorFunction$$1}; /** * Gets / sets the backgroundFill - * (see {@link bar#backgroundFill}) + * (see {@link scatter#backgroundFill}) * @param {string} [_=none] - * @returns {bar | string} - * @memberof bar + * @returns {scatter | string} + * @memberof scatter * @property * by default backgroundFill = 'transparent' */ - bar.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, bar) : backgroundFill; }; + scatter.backgroundFill = function(_) { return arguments.length ? (backgroundFill =_, scatter) : backgroundFill}; /** * Gets / sets the namespace - * (see {@link bar#namespace}) + * (see {@link scatter#namespace}) * @param {string} [_=none] - * @returns {bar | string} - * @memberof bar + * @returns {scatter | string} + * @memberof scatter * @property - * by default namespace = 'd3sm-bar' + * by default namespace = 'd3sm-scatter' */ - bar.namespace = function(_) { return arguments.length ? (namespace = _, bar) : namespace; }; + scatter.namespace = function(_) { return arguments.length ? (namespace =_, scatter) : namespace}; /** * Gets / sets the objectClass - * (see {@link bar#objectClass}) + * (see {@link scatter#objectClass}) * @param {string} [_=none] - * @returns {bar | string} - * @memberof bar + * @returns {scatter | string} + * @memberof scatter * @property * by default objectClass = 'tick-group' */ - bar.objectClass = function(_) { return arguments.length ? (objectClass = _, bar) : objectClass; }; + scatter.objectClass = function(_) { return arguments.length ? (objectClass =_, scatter) : objectClass}; /** * Gets / sets the transitionDuration - * (see {@link bar#transitionDuration}) + * (see {@link scatter#transitionDuration}) * @param {number} [_=none] - * @returns {bar | number} - * @memberof bar + * @returns {scatter | number} + * @memberof scatter * @property * by default transitionDuration = 1000 */ - bar.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, bar) : transitionDuration; }; + scatter.transitionDuration = function(_) { return arguments.length ? (transitionDuration =_, scatter) : transitionDuration}; /** * Gets / sets the easeFunc - * (see {@link bar#easeFunc}) + * (see {@link scatter#easeFunc}) * @param {d3.ease} [_=none] - * @returns {bar | d3.ease} - * @memberof bar + * @returns {scatter | d3.ease} + * @memberof scatter * @property * by default easeFunc = d3.easeExp */ - bar.easeFunc = function(_) { return arguments.length ? (easeFunc = _, bar) : easeFunc; }; - + scatter.easeFunc = function(_) { return arguments.length ? (easeFunc =_, scatter) : easeFunc}; /** - * Gets / sets the barKeys - * (see {@link bar#barKeys}) + * Gets / sets the pointKeys + * (see {@link scatter#pointKeys}) * @param {string[]} [_=none] - * @returns {bar | string[]} - * @memberof bar + * @returns {scatter | string[]} + * @memberof scatter * @property - * by default barKeys = undefined + * by default pointKeys = undefined */ - bar.barKeys = function(_) { return arguments.length ? (barKeys = _, bar) : barKeys; }; + scatter.pointKeys = function(_) { return arguments.length ? (pointKeys =_, scatter) : pointKeys}; /** - * Gets / sets the barValues - * (see {@link bar#barValues}) + * Gets / sets the valuesX + * (see {@link scatter#valuesX}) * @param {number[]} [_=none] - * @returns {bar | number[]} - * @memberof bar + * @returns {scatter | number[]} + * @memberof scatter * @property - * by default barValues = undefined + * by default valuesX = undefined */ - bar.barValues = function(_) { return arguments.length ? (barValues = _, bar) : barValues; }; + scatter.valuesX = function(_) { return arguments.length ? (valuesX =_, scatter) : valuesX}; /** - * Gets / sets the objectSize - * (see {@link bar#objectSize}) - * @param {number} [_=none] - * @returns {bar | number} - * @memberof bar + * Gets / sets the valuesY + * (see {@link scatter#valuesY}) + * @param {number[]} [_=none] + * @returns {scatter | number[]} + * @memberof scatter * @property - * by default objectSize = undefined + * by default valuesY = undefined */ - bar.objectSize = function(_) { return arguments.length ? (objectSize = _, bar) : objectSize; }; + scatter.valuesY = function(_) { return arguments.length ? (valuesY =_, scatter) : valuesY}; /** - * Gets / sets the spacerSize - * (see {@link bar#spacerSize}) - * @param {number} [_=none] - * @returns {bar | number} - * @memberof bar + * Gets / sets the valuesR + * (see {@link scatter#valuesR}) + * @param {number[]} [_=none] + * @returns {scatter | number[]} + * @memberof scatter * @property - * by default spacerSize = undefined + * by default valuesR = undefined */ - bar.spacerSize = function(_) { return arguments.length ? (spacerSize = _, bar) : spacerSize; }; - + scatter.valuesR = function(_) { return arguments.length ? (valuesR =_, scatter) : valuesR}; /** * Gets / sets the tooltip - * (see {@link bar#tooltip}) + * (see {@link scatter#tooltip}) * @param {tooltip} [_=none] - * @returns {bar | tooltip} - * @memberof bar + * @returns {scatter | tooltip} + * @memberof scatter * @property * by default tooltip = tooltip() */ - bar.tooltip = function(_) { return arguments.length ? (tooltip$$1 = _, bar) : tooltip$$1; }; - bar.barPercent = function(_) { return arguments.length ? (barPercent = _, bar) : barPercent; }; + scatter.tooltip = function(_) { return arguments.length ? (tooltip$$1 =_, scatter) : tooltip$$1}; - function bar() { - // for convenience in handling orientation specific values - var horizontalQ = (orient == 'horizontal' || orient == 'bottom' || orient == 'top') ? true : false; - var verticalQ = !horizontalQ; + function scatter() { // background cliping rectangle var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; - var container = setupContainer( selection, namespace, bgcpRect, backgroundFill ); - - // to prevent re-calculation and getters to be passed to axes - barKeys = d3.keys(data); - barValues = barKeys.map(valueExtractor); - - // if grouping is undefined sort barKeys by sortingFunction - var ordered = (grouping == undefined) ? barKeys.sort(sortingFunction) : grouping; - // ordered might be nested depending on grouping - barKeys = flatten(ordered); - - var numberOfObjects = barKeys.length; - var extent = [Math.min(...barValues) - domainPadding,Math.max(...barValues) + domainPadding]; + var container = utils$1.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); + pointKeys = d3.keys(data); + valuesX = pointKeys.map(valueExtractorX); + valuesY = pointKeys.map(valueExtractorY); + valuesR = pointKeys.map(valueExtractorR); - // set the scale + var numberOfObjects = pointKeys.length; + var extentX = [Math.min(...valuesX) - domainPaddingX, Math.max(...valuesX) + domainPaddingX]; + var extentY = [Math.min(...valuesY) - domainPaddingY, Math.max(...valuesY) + domainPaddingY]; + var extentR = [Math.min(...valuesR) - domainPaddingR, Math.max(...valuesR) + domainPaddingR]; - scale.domain(extent).range(horizontalQ - ? [0,spaceY] - : orient == 'right' - ? [0, spaceX] - : [spaceX, 0] - ); - var space = horizontalQ ? spaceX : spaceY; - // calculate object size - objectSize = (objectSize == undefined) - ? calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ) - : objectSize; + scaleX.domain(extentX).range([0, spaceX]); + scaleY.domain(extentY).range([spaceY, 0]); + scaleR.domain(extentR).range([minRadius, maxRadius]); - // calculate spacer size if needed - spacerSize = (spacerSize == undefined) - ? calculateWidthOfSpacer(barKeys, space, objectSize, numberOfObjects, objectSpacer, overflowQ) - : spacerSize; - // make the nested groups - var spacerFunction = groupingSpacer() - .horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects) - .objectClass(objectClass).objectSize(objectSize).spacerSize(spacerSize) - .transitionDuration(transitionDuration).easeFunc(easeFunc) - .namespace(namespace); - // safe default function - var defaultExit = spacerFunction.exitFunction(); + var points = container.selectAll('.'+objectClass); + points = points.data(pointKeys); + var pEnter = points.enter().append('circle') + .attr('class', objectClass) + .attr('cx', 0).attr('cy', spaceY).attr('r', 0); - spacerFunction.exitFunction(function(sel){ - // use default to move objects off screen - // console.log("EXIT", sel.nodes(), objectSize, scale(extent[1])) - if (objectSize == undefined) {console.log(sel.nodes(), objectSize);} - defaultExit(sel); - sel.selectAll('g').classed("to-remove", true); - // shrink rectangles in addition - sel.selectAll('* > rect') - .transition().duration(transitionDuration) - .attr('transform', function(d, i) { - var - x = horizontalQ - ? 0 - : 0 - , - y = verticalQ - ? 0 - : scale(extent[1]) - , - t = 'translate('+x+','+y+')'; - return t - }) - .attr('width', horizontalQ ? objectSize : 0) - .attr('height', verticalQ ? objectSize : 0) - .remove(); - }); + var pExit = points.exit(); + points = points.merge(pEnter); - // move stuff - spacerFunction(container, ordered, 0); + points.each(function(key, i){ + var t = d3.select(this), + currentData = data[key], + x = valuesX[i], + y = valuesY[i], + r = valuesR[i], + fillColor = colorFunction$$1(key, currentData, i, 'fill'), + strokeColor = colorFunction$$1(key, currentData, i, 'stroke'); + t.transition().duration(transitionDuration).ease(easeFunc) + .attr('cx', scaleX(x)) + .attr('cy', scaleY(y)) + .attr('r', scaleR(r)) + .attr('fill', fillColor) + .attr('stroke', strokeColor) + .attr('stroke-width', pointStrokeWidth); + t.on('mouseover', function(d, i){ + points.style('opacity', 0.2); + t.style('opacity', 1); + t.transition().duration(transitionDuration/2).ease(easeFunc) + .attr('stroke-width', pointStrokeWidth*2) + .attr('r', scaleR(r) * 1.5); - var parentIndexArray = []; - container.selectAll('g:not(.to-remove).'+objectClass) - .each(function(d, i){parentIndexArray.push(Number(d3.select(this).attr('parent-index')));}); - - - colorFunction$$1 = colorFunction$$1.colorBy() == 'index' - ? colorFunction$$1.dataExtent([0, Math.max(...parentIndexArray)]) - : colorFunction$$1.dataExtent(extent); - - - - container.selectAll('g.'+objectClass+':not(.to-remove)').each(function(key, i) { - // console.log(key, scale(extent[1]) - scale(valueExtractor(key, i))) - var t = d3.select(this), - currentData = data[key], - value = valueExtractor(key, i), - i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'), - fillColor = colorFunction$$1(key, value, i, 'fill'), // prevent duplicate computation - strokeColor = colorFunction$$1(key, value, i, 'stroke'); - - - var bar = safeSelect(t, 'rect', 'bar-rect'); - - if (bar.attr('transform') == undefined) { - bar.attr('transform', function(d, i) { - var - x = horizontalQ - ? 0 - : 0 - , - y = verticalQ - ? 0 - : scale(extent[1]) - , - t = 'translate('+x+','+y+')'; - return t - }) - .attr('width', horizontalQ ? objectSize : 0) - .attr('height', verticalQ ? objectSize : 0); - - } - + }); + t.node().addEventListener('mouseout', function(){ + container.selectAll('.'+objectClass).style('opacity', 1); + t.transition().duration(transitionDuration/2).ease(easeFunc) + .attr('stroke-width', pointStrokeWidth) + .attr('r', scaleR(r)); - bar.transition().duration(transitionDuration).ease(easeFunc) - .attr('transform', function(d, i) { - var - x = horizontalQ - ? objectSize - objectSize * barPercent - : orient == 'right' - ? scale(extent[1]) - scale(value) - : objectSize - objectSize * barPercent - , - y = verticalQ - ? objectSize - objectSize * barPercent - : scale(extent[1]) - scale(value) - , - t = 'translate('+x+','+y+')'; - return t - }) - .attr('width', horizontalQ ? objectSize * barPercent : scale(value)) - .attr('height', verticalQ ? objectSize * barPercent: scale(value)) - .attr('fill', fillColor) - .attr('stroke', strokeColor) - .attr('stroke-width', barStrokeWidth); + }); + }); - t.on('mouseover', function(d, i){ - container.selectAll('g.'+objectClass).style('opacity', 0.2); - t.style('opacity', 1); - bar.attr('stroke-width',barStrokeWidth*2); - }); - t.on('mouseout', function(){ - container.selectAll('g.'+objectClass).style('opacity', 1); - bar.attr('stroke-width', barStrokeWidth); - }); - }); + pExit.transition().duration(transitionDuration).ease(easeFunc) + .attr('cx', 0).attr('cy', spaceY).attr('r', 0) + .remove(); - tooltip$$1.selection(container.selectAll('.bar-rect')) + tooltip$$1.selection(points) .data(data); tooltip$$1(); - } - return bar + + + return scatter } +/******************************************************************************* +** ** +** ** +** BAR ** +** ** +** ** +*******************************************************************************/ + /** - * Creates a bubbleHeatmap + * Creates a bar * - * {@link https://sumneuron.gitlab.io/d3sm/demos/bubble-heatmap/index.html Demo} - * @constructor bubbleHeatmap + * {@link https://sumneuron.gitlab.io/d3sm/demos/bar-chart-same-data-complex-grouping/index.html Demo} + * @constructor bar * @param {d3.selection} selection - * @namespace bubbleHeatmap - * @returns {function} bubbleHeatmap + * @namespace bar + * @returns {function} bar */ -function bubbleHeatmap( selection ) { - var +function bar ( selection ) { + /* + Assumes that data is list an object. + + The keys of data will be bound to the bars. + The valueExtractor function will extract the value from data[key]. + + Grouping can be used if desired. It should be an arbitrary complex list where + the values are strings matching keys in data. + */ + var /** - * Data to plot. Assumed to be a object, where each key corresponds to a cell - * (see {@link bubbleHeatmap#data}) + * Data to plot. Assumed to be a object, where each key corresponds to a bar + * (see {@link bar#data}) * @param {Object} [data=undefined] - * @memberof bubbleHeatmap# + * @memberof bar# * @property */ data, - /** - * Amount of horizontal space (in pixels) avaible to render the bubbleHeatmap in - * (see {@link bubbleHeatmap#spaceX}) + * Which direction to render the bars in + * (see {@link bar#orient}) + * @param {number} [orient='horizontal'] + * @memberof bar# + * @property + */ + orient='horizontal', + /** + * Amount of horizontal space (in pixels) avaible to render the bar in + * (see {@link bar#spaceX}) * @param {number} [spaceX=undefined] - * @memberof bubbleHeatmap# + * @memberof bar# * @property */ spaceX, /** - * Amount of vertical space (in pixels) avaible to render the bubbleHeatmap in - * (see {@link bubbleHeatmap.spaceY}) + * Amount of vertical space (in pixels) avaible to render the bar in + * (see {@link bar.spaceY}) * @param {number} [spaceY=undefined] - * @memberof bubbleHeatmap# + * @memberof bar# * @property */ spaceY, /** - * The internal key of the cell specifiying to which x axis key it belongs - * (see {@link bubbleHeatmap.xKey}) - * @param {string} [xKey='x'] - * @memberof bubbleHeatmap# + * Whether or not to allow bar to render elements pass the main spatial dimension + * given the orientation (see {@link bar#orient}), where {@link bar#orient}="horizontal" + * the main dimension is {@link bar#spaceX} and where {@link bar#orient}="vertical" + * the main dimension is {@link bar#spaceY} + * @param {boolean} [overflowQ=false] + * @memberof bar# * @property */ - xKey = 'x', + overflowQ = false, + /** - * The internal key of the cell specifiying to which y axis key it belongs - * (see {@link bubbleHeatmap.yKey}) - * @param {string} [yKey='y'] - * @memberof bubbleHeatmap# + * An array - putatively of other arrays - depicting how bars should be arranged + * @param {Array[]} [grouping=undefined] + * @memberof bar# * @property */ - yKey = 'y', + grouping, + /** - * The internal key of the cell specifiying what value to use to determine the radius - * (see {@link bubbleHeatmap.rKey}) - * @param {string} [rKey='r'] - * @memberof bubbleHeatmap# + * How to get the value of the bar + * @param {function} [valueExtractor=function(key, index) { return data[key] }] + * @memberof bar# * @property */ - rKey = 'r', + valueExtractor = function(key, index) { return data[key] }, /** - * The internal key of the cell specifiying what value to use to determine the color - * (see {@link bubbleHeatmap.vKey}) - * @param {string} [vKey='v'] - * @memberof bubbleHeatmap# + * How to sort the bars - if {@link bar#grouping} is not provided. + * @param {function} [sortingFunction=function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}] + * @memberof bar# * @property */ - vKey = 'v', + sortingFunction = function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}, /** - * Function for extracting the the value from xKey. - * (see {@link bubbleHeatmap.xExtractor}) - * @param {function} [xExtractor=function(key, i) { return data[key][xKey] }] - * @returns {string} - * @memberof bubbleHeatmap# + * The scale for which bar values should be transformed by + * @param {d3.scale} [scale=d3.scaleLinear] + * @memberof bar# * @property */ - xExtractor = function(key, i) {return data[key][xKey] }, + scale = d3.scaleLinear(), /** - * Function for extracting the the value from yKey. - * (see {@link bubbleHeatmap.yExtractor}) - * @param {function} [yExtractor=function(key, i) { return data[key][yKey] }] - * @returns {string} - * @memberof bubbleHeatmap# + * The padding for the domain of the scale (see {@link bar#scale}) + * @param {number} [domainPadding=0.5] + * @memberof bar# * @property */ - yExtractor = function(key, i) { return data[key][yKey] }, + domainPadding = 0.5, + /** - * Function for extracting the the value from rKey. - * (see {@link bubbleHeatmap.rExtractor}) - * @param {function} [rExtractor=function(key, i) { return data[key][rKey] }] - * @returns {number} - * @memberof bubbleHeatmap# + * Default space for the spacer (percentage) of main dimension given the orientation + * (see {@link bar#orient}), where {@link bar#orient}="horizontal" + * the main dimension is {@link bar#spaceX} and where {@link bar#orient}="vertical" + * the main dimension is {@link bar#spaceY} between bars + * @param {number} [objectSpacer=0.05] + * @memberof bar# * @property */ - rExtractor = function(key, i) { return data[key][rKey] }, + objectSpacer = 0.05, /** - * Function for extracting the the value from vKey. - * (see {@link bubbleHeatmap.vExtractor}) - * @param {function} [vExtractor=function(key, i) { return data[key][vKey] }] - * @returns {number} - * @memberof bubbleHeatmap# + * The minimum size that an object can be + * @param {number} [minObjectSize=50] + * @memberof bar# * @property */ - vExtractor = function(key, i) { return data[key][vKey] }, - - + minObjectSize = 50, /** - * Whether or not to allow bubbleHeatmap to render elements pass the main spatial dimension - * given the orientation (see {@link bubbleHeatmap#orient}), where {@link bubbleHeatmap#orient}="bottom" or {@link bubbleHeatmap#orient}="top" - * the main dimension is {@link bubbleHeatmap#spaceX} and where {@link bubbleHeatmap#orient}="left" or {@link bubbleHeatmap#orient}="right" - * the main dimension is {@link bubbleHeatmap#spaceY} - * @param {boolean} [overflowQ=false] - * @memberof bubbleHeatmap# + * The maximum size that an object can be + * @param {number} [maxObjectSize=100] + * @memberof bar# * @property */ - overflowQ = false, + maxObjectSize = 100, /** - * The scale for which the radius values should be transformed by - * @param {d3.scale} [scale=d3.scaleLinear] - * @memberof bubbleHeatmap# + * The stroke width of the bars + * @param {number} [barStrokeWidth=2] + * @memberof bar# * @property */ - scale = d3.scaleLinear(), + barStrokeWidth = 2, /** - * The padding for the domain of the scale (see {@link bubbleHeatmap#scale}) - * @param {number} [domainPadding=0.5] - * @memberof bubbleHeatmap# - * @property - */ - domainPadding = 0.5, - - /** - * Default space for the spacer (percentage) of main dimension given the orientation - * (see {@link bubbleHeatmap#orient}), where {@link bubbleHeatmap#orient}="horizontal" - * the main dimension is {@link bubbleHeatmap#spaceX} and where {@link bubbleHeatmap#orient}="vertical" - * the main dimension is {@link bubbleHeatmap#spaceY} between bubbles - * @param {number} [objectSpacer=0.0] - * @memberof bubbleHeatmap# - * @property - */ - objectSpacer = 0.0, - /** - * The minimum size that an object can be - * @param {number} [minObjectSize=50] - * @memberof bubbleHeatmap# - * @property - */ - minObjectSize = 50, - /** - * The maximum size that an object can be - * @param {number} [maxObjectSize=100] - * @memberof bubbleHeatmap# + * Instance of ColorFunction + * @param {function} [colorFunction = colorFunction()] + * @memberof bar# * @property */ - maxObjectSize = 100, - + colorFunction$$1 = colorFunction(), - /** - * The stroke width of the bubbles - * @param {number} [bubbleStrokeWidth=2] - * @memberof bubbleHeatmap# - * @property - */ - bubbleStrokeWidth = 2, - // colorFunc = colorFunction(), /** * Color of the background * @param {string} [backgroundFill="transparent"] - * @memberof bubbleHeatmap# + * @memberof bar# * @property */ backgroundFill = 'transparent', /** - * Namespace for all items made by this instance of bubbleHeatmap - * @param {string} [namespace="d3sm-bubble"] - * @memberof bubbleHeatmap# + * Namespace for all items made by this instance of bar + * @param {string} [namespace="d3sm-bar"] + * @memberof bar# * @property */ - namespace = 'd3sm-bubble', + namespace = 'd3sm-bar', /** - * Class name for bubble container ( element) - * @param {string} [objectClass="bubble"] - * @memberof bubbleHeatmap# + * Class name for bar container ( element) + * @param {string} [objectClass="bar"] + * @memberof bar# * @property */ - objectClass = 'bubble', + objectClass = 'bar', + /** * Duration of all transitions of this element * @param {number} [transitionDuration=1000] - * @memberof bubbleHeatmap# + * @memberof bar# * @property */ transitionDuration = 1000, /** * Easing function for transitions * @param {d3.ease} [easeFunc=d3.easeExp] - * @memberof bubbleHeatmap# + * @memberof bar# * @property */ easeFunc = d3.easeExp, + // useful values to extract to prevent re-calculation /** - * Stores the keys of all the cells - * Calculated after bubbleHeatmap called. - * @param {string[]} [cellKeys=undefined] - * @memberof bubbleHeatmap# - * @property - */ - cellKeys, - /** - * Stores the list of unique xValues - * Calculated after bubbleHeatmap called. - * @param {string[]} [xValues=undefined] - * @memberof bubbleHeatmap# - * @property - */ - xValues, - /** - * Stores the list of unique yValues - * Calculated after bubbleHeatmap called. - * @param {string[]} [yValues=undefined] - * @memberof bubbleHeatmap# + * The keys of the bars + * @param {string[]} [barKeys=undefined] + * @memberof bar# * @property */ - yValues, + barKeys, /** - * Stores the list of unique rValues - * Calculated after bubbleHeatmap called. - * @param {string[]} [rValues=undefined] - * @memberof bubbleHeatmap# + * The values of the bars + * @param {number[]} [barValues=undefined] + * @memberof bar# * @property */ - rValues, + barValues, /** - * Stores the list of unique vValues - * Calculated after bubbleHeatmap called. - * @param {string[]} [vValues=undefined] - * @memberof bubbleHeatmap# + * The objectSize (actual width) used by the bars + * @param {number} [objectSize=undefined] + * @memberof bar# * @property */ - vValues, - - xKeySortingFunction = function(a, b) { return xExtractor(a) - xExtractor(b) }, - yKeySortingFunction = function(a, b) { return yExtractor(a) - yExtractor(b) }, + objectSize, /** - * Instance of ColorFunction with .colorBy set to 'category' - * @function colorFunction - * @memberof bubbleHeatmap# + * The spacerSize (actual width) used by the spacers between the bars + * @param {number} [spacerSize=undefined] + * @memberof bar# * @property */ - colorFunction$$1 = colorFunction().colorBy('value'), + spacerSize, /** * Instance of Tooltip - * @function tooltip - * @memberof bubbleHeatmap# + * @param {function} [tooltip=tooltip()] + * @memberof bar# * @property */ tooltip$$1 = tooltip(), - - /** - * store the size the bubble could be in the x dimension - * the actuall size of the bubble will be the min of xSize and {@link bubbleHeatmap#ySize} - * Calculated after bubbleHeatmap called. - * @param {string[]} [xSize=undefined] - * @memberof bubbleHeatmap# - * @property - */ - xSize, - /** - * store the size of the spacer in the x dimension - * Calculated after bubbleHeatmap called. - * @param {string[]} [xSpacerSize=undefined] - * @memberof bubbleHeatmap# - * @property - */ - xSpacerSize, - - /** - * store the size the bubble could be in the y dimension - * the actuall size of the bubble will be the min of xSize and {@link bubbleHeatmap#xSize} - * Calculated after bubbleHeatmap called. - * @param {string[]} [ySize=undefined] - * @memberof bubbleHeatmap# - * @property - */ - ySize, - /** - * store the size of the spacer in the y dimension. - * Calculated after bubbleHeatmap called. - * @param {string[]} [xSpacerSize=undefined] - * @memberof bubbleHeatmap# - * @property - */ - ySpacerSize; + barPercent = 1; /** * Gets or sets the selection in which items are manipulated * @param {d3.selection} [_=none] - * @returns {bubbleHeatmap | d3.selection} - * @memberof bubbleHeatmap + * @returns {bar | d3.selection} + * @memberof bar * @property * by default selection = selection */ - bhm.selection = function(_) { return arguments.length ? (selection = _, bhm) : selection; }; + bar.selection = function(_) { return arguments.length ? (selection = _, bar) : selection; }; /** * Gets or sets the data - * (see {@link bubbleHeatmap#data}) + * (see {@link bar#data}) * @param {number} [_=none] - * @returns {bubbleHeatmap | object} - * @memberof bubbleHeatmap + * @returns {bar | object} + * @memberof bar * @property */ - bhm.data = function(_) { return arguments.length ? (data = _, bhm) : data; }; - // bhm.orient = function(_) { return arguments.length ? (orient = _, bhm) : orient; } + bar.data = function(_) { return arguments.length ? (data = _, bar) : data; }; + /** + * Gets or sets the orient of the bars + * (see {@link bar#orient}) + * @param {number} [_=none] + * @returns {bar | object} + * @memberof bar + * @property + */ + bar.orient = function(_) { return arguments.length ? (orient = _, bar) : orient; }; /** * Gets or sets the amount of horizontal space in which items are manipulated - * (see {@link bubbleHeatmap#spaceX}) + * (see {@link bar#spaceX}) * @param {number} [_=none] should be a number > 0 - * @returns {bubbleHeatmap | number} - * @memberof bubbleHeatmap + * @returns {bar | number} + * @memberof bar * @property * by default spaceX = undefined */ - bhm.spaceX = function(_) { return arguments.length ? (spaceX = _, bhm) : spaceX; }; + bar.spaceX = function(_) { return arguments.length ? (spaceX = _, bar) : spaceX; }; /** * Gets or sets the amount of vertical space in which items are manipulated - * (see {@link bubbleHeatmap#spaceY}) + * (see {@link bar#spaceY}) * @param {number} [_=none] should be a number > 0 - * @returns {bubbleHeatmap | number} - * @memberof bubbleHeatmap + * @returns {bar | number} + * @memberof bar * @property * by default spaceY = undefined */ - bhm.spaceY = function(_) { return arguments.length ? (spaceY = _, bhm) : spaceY; }; + bar.spaceY = function(_) { return arguments.length ? (spaceY = _, bar) : spaceY; }; /** - * Gets or sets the xKey - * (see {@link bubbleHeatmap#xKey}) - * @param {string} [_=none] - * @returns {bubbleHeatmap | string} - * @memberof bubbleHeatmap + * Gets / sets whether or not bar is allowed to go beyond specified dimensions + * (see {@link bar#spaceX}) + * @param {boolean} [_=none] + * @returns {bar | boolean} + * @memberof bar * @property - * by default xKey = 'x' + * by default overflowQ = false */ - bhm.xKey = function(_) { return arguments.length ? (xKey = _, bhm) : xKey; }; + bar.overflowQ = function(_) { return arguments.length ? (overflowQ = _, bar) : overflowQ; }; /** - * Gets or sets the yKey - * (see {@link bubbleHeatmap#yKey}) - * @param {string} [_=none] - * @returns {bubbleHeatmap | string} - * @memberof bubbleHeatmap + * Gets / sets the grouping of the bars + * (see {@link bar#grouping}) + * @param {Array[]} [_=none] + * @returns {bar | Array[]} + * @memberof bar * @property - * by default yKey = 'y' + * by default grouping = undefined */ - bhm.yKey = function(_) { return arguments.length ? (yKey = _, bhm) : yKey; }; + bar.grouping = function(_) { return arguments.length ? (grouping = _, bar) : grouping; }; /** - * Gets or sets the rKey - * (see {@link bubbleHeatmap#rKey}) - * @param {string} [_=none] - * @returns {bubbleHeatmap | string} - * @memberof bubbleHeatmap + * Gets / sets the valueExtractor + * (see {@link bar#valueExtractor}) + * @param {function} [_=none] + * @returns {bar | function} + * @memberof bar * @property - * by default rKey = 'r' + * by default valueExtractor = function(key, index) { return data[key] }, */ - bhm.rKey = function(_) { return arguments.length ? (rKey = _, bhm) : rKey; }; + bar.valueExtractor = function(_) { return arguments.length ? (valueExtractor = _, bar) : valueExtractor; }; /** - * Gets or sets the vKey - * (see {@link bubbleHeatmap#vKey}) - * @param {string} [_=none] - * @returns {bubbleHeatmap | string} - * @memberof bubbleHeatmap + * Gets / sets the sortingFunction + * (see {@link bar#sortingFunction}) + * @param {function} [_=none] + * @returns {bar | function} + * @memberof bar * @property - * by default vKey = 'y' + * by default sortingFunction = function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}, */ - bhm.vKey = function(_) { return arguments.length ? (vKey = _, bhm) : vKey; }; - + bar.sortingFunction = function(_) { return arguments.length ? (sortingFunction = _, bar) : sortingFunction; }; /** - * Gets or sets the cellKeys - * (see {@link bubbleHeatmap#cellKeys}) - * @param {string[]} [_=none] - * @returns {bubbleHeatmap | string[]} - * @memberof bubbleHeatmap - * @property - * by default cellKeys = undefined - */ - bhm.cellKeys = function(_) { return arguments.length ? (cellKeys = _, bhm) : cellKeys; }; - /** - * Gets or sets the xValues - * (see {@link bubbleHeatmap#xValues}) - * @param {string[]} [_=none] - * @returns {bubbleHeatmap | string[]} - * @memberof bubbleHeatmap - * @property - * by default xValues = undefined - */ - bhm.xValues = function(_) { return arguments.length ? (xValues = _, bhm) : xValues; }; - /** - * Gets or sets the yValues - * (see {@link bubbleHeatmap#yValues}) - * @param {string[]} [_=none] - * @returns {bubbleHeatmap | string[]} - * @memberof bubbleHeatmap - * @property - * by default yValues = undefined - */ - bhm.yValues = function(_) { return arguments.length ? (yValues = _, bhm) : yValues; }; - /** - * Gets or sets the rValues - * (see {@link bubbleHeatmap#rValues}) - * @param {number[]} [_=none] - * @returns {bubbleHeatmap | number[]} - * @memberof bubbleHeatmap - * @property - * by default rValues = undefined - */ - bhm.rValues = function(_) { return arguments.length ? (rValues = _, bhm) : rValues; }; - /** - * Gets or sets the vValues - * (see {@link bubbleHeatmap#vValues}) - * @param {number[]} [_=none] - * @returns {bubbleHeatmap | number[]} - * @memberof bubbleHeatmap - * @property - * by default vValues = undefined - */ - bhm.vValues = function(_) { return arguments.length ? (vValues = _, bhm) : vValues; }; - - - /** - * Gets or sets the xExtractor - * (see {@link bubbleHeatmap#xExtractor}) - * @param {function} [_=none] - * @returns {bubbleHeatmap | function} - * @memberof bubbleHeatmap - * @property - * by default xExtractor = undefined - */ - bhm.xExtractor = function(_) { return arguments.length ? (xExtractor = _, bhm) : xExtractor; }; - /** - * Gets or sets the yExtractor - * (see {@link bubbleHeatmap#yExtractor}) - * @param {function} [_=none] - * @returns {bubbleHeatmap | function} - * @memberof bubbleHeatmap - * @property - * by default yExtractor = undefined - */ - bhm.yExtractor = function(_) { return arguments.length ? (yExtractor = _, bhm) : yExtractor; }; - /** - * Gets or sets the rExtractor - * (see {@link bubbleHeatmap#rExtractor}) - * @param {function} [_=none] - * @returns {bubbleHeatmap | function} - * @memberof bubbleHeatmap - * @property - * by default rExtractor = undefined - */ - bhm.rExtractor = function(_) { return arguments.length ? (rExtractor = _, bhm) : rExtractor; }; - /** - * Gets or sets the vExtractor - * (see {@link bubbleHeatmap#vExtractor}) - * @param {function} [_=none] - * @returns {bubbleHeatmap | function} - * @memberof bubbleHeatmap - * @property - * by default vExtractor = undefined - */ - bhm.vExtractor = function(_) { return arguments.length ? (vExtractor = _, bhm) : vExtractor; }; - - /** - * Gets / sets whether or not bubbleHeatmap is allowed to go beyond specified dimensions - * (see {@link bubbleHeatmap#spaceX}) - * @param {boolean} [_=none] - * @returns {bubbleHeatmap | boolean} - * @memberof bubbleHeatmap - * @property - * by default overflowQ = false - */ - bhm.overflowQ = function(_) { return arguments.length ? (overflowQ = _, bhm) : overflowQ; }; - /** - * Gets / sets the scale for which the radius of bubbles should be transformed by - * (see {@link bubbleHeatmap#scale}) + * Gets / sets the scale for which the bar values should be transformed by + * (see {@link bar#scale}) * @param {d3.scale} [_=none] - * @returns {bubbleHeatmap | d3.scale} - * @memberof bubbleHeatmap + * @returns {bar | d3.scale} + * @memberof bar * @property * by default scale = d3.scaleLinear() */ - bhm.scale = function(_) { return arguments.length ? (scale = _, bhm) : scale; }; + bar.scale = function(_) { return arguments.length ? (scale = _, bar) : scale; }; /** * Gets / sets the padding for the domain of the scale - * (see {@link bubbleHeatmap#domainPadding}) + * (see {@link bar#domainPadding}) * @param {number} [_=none] - * @returns {bubbleHeatmap | number} - * @memberof bubbleHeatmap + * @returns {bar | number} + * @memberof bar * @property * by default domainPadding = 0.5 */ - bhm.domainPadding = function(_) { return arguments.length ? (domainPadding = _, bhm) : domainPadding; }; + bar.domainPadding = function(_) { return arguments.length ? (domainPadding = _, bar) : domainPadding; }; /** * Gets / sets objectSpacer - * (see {@link bubbleHeatmap#objectSpacer}) + * (see {@link bar#objectSpacer}) * @param {number} [_=none] - * @returns {bubbleHeatmap | number} - * @memberof bubbleHeatmap + * @returns {bar | number} + * @memberof bar * @property - * by default objectSpacer = 0.0 + * by default objectSpacer = 0.05 */ - bhm.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, objectSpacer) : data; }; + bar.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, bar) : objectSpacer; }; /** * Gets / sets the minObjectSize - * (see {@link bubbleHeatmap#minObjectSize}) + * (see {@link bar#minObjectSize}) * @param {number} [_=none] - * @returns {bubbleHeatmap | number} - * @memberof bubbleHeatmap + * @returns {bar | number} + * @memberof bar * @property * by default minObjectSize = 50 */ - bhm.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, bhm) : minObjectSize; }; + bar.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, bar) : minObjectSize; }; /** * Gets / sets the maxObjectSize - * (see {@link bubbleHeatmap#maxObjectSize}) + * (see {@link bar#maxObjectSize}) * @param {number} [_=none] - * @returns {bubbleHeatmap | number} - * @memberof bubbleHeatmap + * @returns {bar | number} + * @memberof bar * @property * by default maxObjectSize = 100 */ - bhm.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, bhm) : maxObjectSize; }; + bar.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, bar) : maxObjectSize; }; + /** - * Gets / sets the bubbleStrokeWidth - * (see {@link bubbleHeatmap#bubbleStrokeWidth}) + * Gets / sets the barStrokeWidth + * (see {@link bar#barStrokeWidth}) * @param {number} [_=none] - * @returns {bubbleHeatmap | number} - * @memberof bubbleHeatmap + * @returns {bar | number} + * @memberof bar * @property - * by default bubbleStrokeWidth = 2 + * by default barStrokeWidth = 2 */ - bhm.bubbleStrokeWidth = function(_) { return arguments.length ? (bubbleStrokeWidth = _, bhm) : bubbleStrokeWidth; }; + bar.barStrokeWidth = function(_) { return arguments.length ? (barStrokeWidth = _, bar) : barStrokeWidth; }; + /** + * Gets / sets the colorFunction + * (see {@link bar#colorFunction}) + * @param {number} [_=none] + * @returns {bar | number} + * @memberof bar + * @property + * by default colorFunction = colorFunction() + */ + bar.colorFunction = function(_) { return arguments.length ? (colorFunction$$1 = _, bar) : colorFunction$$1; }; + /** * Gets / sets the backgroundFill - * (see {@link bubbleHeatmap#backgroundFill}) + * (see {@link bar#backgroundFill}) * @param {string} [_=none] - * @returns {bubbleHeatmap | string} - * @memberof bubbleHeatmap + * @returns {bar | string} + * @memberof bar * @property * by default backgroundFill = 'transparent' */ - bhm.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, bhm) : backgroundFill; }; + bar.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, bar) : backgroundFill; }; /** * Gets / sets the namespace - * (see {@link bubbleHeatmap#namespace}) + * (see {@link bar#namespace}) * @param {string} [_=none] - * @returns {bubbleHeatmap | string} - * @memberof bubbleHeatmap + * @returns {bar | string} + * @memberof bar * @property - * by default namespace = 'd3sm-bubbleHeatmap' + * by default namespace = 'd3sm-bar' */ - bhm.namespace = function(_) { return arguments.length ? (namespace = _, bhm) : namespace; }; + bar.namespace = function(_) { return arguments.length ? (namespace = _, bar) : namespace; }; /** * Gets / sets the objectClass - * (see {@link bubbleHeatmap#objectClass}) + * (see {@link bar#objectClass}) * @param {string} [_=none] - * @returns {bubbleHeatmap | string} - * @memberof bubbleHeatmap + * @returns {bar | string} + * @memberof bar * @property * by default objectClass = 'tick-group' */ - bhm.objectClass = function(_) { return arguments.length ? (objectClass = _, bhm) : objectClass; }; + bar.objectClass = function(_) { return arguments.length ? (objectClass = _, bar) : objectClass; }; /** * Gets / sets the transitionDuration - * (see {@link bubbleHeatmap#transitionDuration}) + * (see {@link bar#transitionDuration}) * @param {number} [_=none] - * @returns {bubbleHeatmap | number} - * @memberof bubbleHeatmap + * @returns {bar | number} + * @memberof bar * @property * by default transitionDuration = 1000 */ - bhm.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, bhm) : transitionDuration; }; + bar.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, bar) : transitionDuration; }; /** * Gets / sets the easeFunc - * (see {@link bubbleHeatmap#easeFunc}) + * (see {@link bar#easeFunc}) * @param {d3.ease} [_=none] - * @returns {bubbleHeatmap | d3.ease} - * @memberof bubbleHeatmap + * @returns {bar | d3.ease} + * @memberof bar * @property * by default easeFunc = d3.easeExp */ - bhm.easeFunc = function(_) { return arguments.length ? (easeFunc = _, bhm) : easeFunc; }; + bar.easeFunc = function(_) { return arguments.length ? (easeFunc = _, bar) : easeFunc; }; + /** - * Gets / sets the tooltip - * (see {@link bubbleHeatmap#tooltip}) - * @param {tooltip} [_=none] - * @returns {bubbleHeatmap | tooltip} - * @memberof bubbleHeatmap + * Gets / sets the barKeys + * (see {@link bar#barKeys}) + * @param {string[]} [_=none] + * @returns {bar | string[]} + * @memberof bar * @property - * by default tooltip = tooltip() + * by default barKeys = undefined */ - bhm.tooltip = function(_) { return arguments.length ? (tooltip$$1 = _, bhm) : tooltip$$1; }; - + bar.barKeys = function(_) { return arguments.length ? (barKeys = _, bar) : barKeys; }; /** - * Gets / sets the colorFunction - * (see {@link bubbleHeatmap#colorFunction}) - * @param {colorFunction} [_=none] - * @returns {bubbleHeatmap | colorFunction} - * @memberof bubbleHeatmap + * Gets / sets the barValues + * (see {@link bar#barValues}) + * @param {number[]} [_=none] + * @returns {bar | number[]} + * @memberof bar * @property - * by default colorFunction = colorFunction() + * by default barValues = undefined */ - bhm.colorFunction = function(_) { return arguments.length ? (colorFunction$$1 = _, bhm) : colorFunction$$1; }; - + bar.barValues = function(_) { return arguments.length ? (barValues = _, bar) : barValues; }; /** - * Gets / sets the xSize - * (see {@link bubbleHeatmap#xSize}) + * Gets / sets the objectSize + * (see {@link bar#objectSize}) * @param {number} [_=none] - * @returns {bubbleHeatmap | number} - * @memberof bubbleHeatmap + * @returns {bar | number} + * @memberof bar * @property - * by default xSize = undefined + * by default objectSize = undefined */ - bhm.xSize = function(_) { return arguments.length ? (xSize = _, bhm) : xSize; }; + bar.objectSize = function(_) { return arguments.length ? (objectSize = _, bar) : objectSize; }; /** - * Gets / sets the xSpacerSize - * (see {@link bubbleHeatmap#xSpacerSize}) + * Gets / sets the spacerSize + * (see {@link bar#spacerSize}) * @param {number} [_=none] - * @returns {bubbleHeatmap | number} - * @memberof bubbleHeatmap - * @property - * by default xSpacerSize = undefined - */ - bhm.xSpacerSize = function(_) { return arguments.length ? (xSpacerSize = _, bhm) : xSpacerSize; }; - /** - * Gets / sets the ySize - * (see {@link bubbleHeatmap#ySize}) - * @param {number} [_=none] - * @returns {bubbleHeatmap | number} - * @memberof bubbleHeatmap + * @returns {bar | number} + * @memberof bar * @property - * by default ySize = undefined + * by default spacerSize = undefined */ - bhm.ySize = function(_) { return arguments.length ? (ySize = _, bhm) : ySize; }; + bar.spacerSize = function(_) { return arguments.length ? (spacerSize = _, bar) : spacerSize; }; + /** - * Gets / sets the ySpacerSize - * (see {@link bubbleHeatmap#ySpacerSize}) - * @param {number} [_=none] - * @returns {bubbleHeatmap | number} - * @memberof bubbleHeatmap + * Gets / sets the tooltip + * (see {@link bar#tooltip}) + * @param {tooltip} [_=none] + * @returns {bar | tooltip} + * @memberof bar * @property - * by default ySpacerSize = undefined + * by default tooltip = tooltip() */ - bhm.ySpacerSize = function(_) { return arguments.length ? (ySpacerSize = _, bhm) : ySpacerSize; }; - // bhm.yKeySortingFunction = function(_) { return arguments.length ? (yKeySortingFunction = _, bhm) : yKeySortingFunction; } - // bhm.xKeySortingFunction = function(_) { return arguments.length ? (xKeySortingFunction = _, bhm) : xKeySortingFunction; } + bar.tooltip = function(_) { return arguments.length ? (tooltip$$1 = _, bar) : tooltip$$1; }; + bar.barPercent = function(_) { return arguments.length ? (barPercent = _, bar) : barPercent; }; + function bar() { + // for convenience in handling orientation specific values + var horizontalQ = (orient == 'horizontal' || orient == 'bottom' || orient == 'top') ? true : false; + var verticalQ = !horizontalQ; - function bhm() { + // background cliping rectangle var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; - var container = setupContainer( selection, namespace, bgcpRect, backgroundFill ); - - cellKeys = d3.keys(data); - cellKeys.sort(function(a, b){ return xKeySortingFunction(a, b) || yKeySortingFunction(a, b) }); - log('bubbleHeatmap', 'cells are sorted by', cellKeys); + var container = utils$1.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); + // to prevent re-calculation and getters to be passed to axes + barKeys = d3.keys(data); + barValues = barKeys.map(valueExtractor); + // if grouping is undefined sort barKeys by sortingFunction + var ordered = (grouping == undefined) ? barKeys.sort(sortingFunction) : grouping; + // ordered might be nested depending on grouping + barKeys = utils$1.arr.flatten(ordered); - xValues = unique(cellKeys.map(xExtractor)); - yValues = unique(cellKeys.map(yExtractor)); - rValues = unique(cellKeys.map(rExtractor)); - vValues = unique(cellKeys.map(vExtractor)); - log('bubbleHeatmap', 'x and y keys are', {x: xValues, y:yValues}); + var numberOfObjects = barKeys.length; + var extent = [Math.min(...barValues) - domainPadding,Math.max(...barValues) + domainPadding]; - var xDim = xValues.length, yDim = yValues.length; + // set the scale - var extent = [Math.min(...rValues) - domainPadding,Math.max(...rValues) + domainPadding]; + scale.domain(extent).range(horizontalQ + ? [0,spaceY] + : orient == 'right' + ? [0, spaceX] + : [spaceX, 0] + ); + var space = horizontalQ ? spaceX : spaceY; + // calculate object size + objectSize = (objectSize == undefined) + ? utils$1.math.calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ) + : objectSize; + // calculate spacer size if needed + spacerSize = (spacerSize == undefined) + ? utils$1.math.calculateWidthOfSpacer(barKeys, space, objectSize, numberOfObjects, objectSpacer, overflowQ) + : spacerSize; + // make the nested groups + var spacerFunction = groupingSpacer() + .horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects) + .objectClass(objectClass).objectSize(objectSize).spacerSize(spacerSize) + .transitionDuration(transitionDuration).easeFunc(easeFunc) + .namespace(namespace); + // safe default function + var defaultExit = spacerFunction.exitFunction(); - ySize = calculateWidthOfObject(spaceY, yDim, minObjectSize, maxObjectSize, objectSpacer, overflowQ); - xSize = calculateWidthOfObject(spaceX, xDim, minObjectSize, maxObjectSize, objectSpacer, overflowQ); - ySpacerSize = calculateWidthOfSpacer(yValues, spaceY, ySize, yDim, objectSpacer, overflowQ); - xSpacerSize = calculateWidthOfSpacer(xValues, spaceX, xSize, xDim, objectSpacer, overflowQ); - log('bubbleHeatmap', 'size of', {x: xSize, y: ySize}); + spacerFunction.exitFunction(function(sel$$1){ + // use default to move objects off screen + // console.log("EXIT", sel.nodes(), objectSize, scale(extent[1])) + if (objectSize == undefined) {console.log(sel$$1.nodes(), objectSize);} + defaultExit(sel$$1); + sel$$1.selectAll('g').classed("to-remove", true); + // shrink rectangles in addition + sel$$1.selectAll('* > rect') + .transition().duration(transitionDuration) + .attr('transform', function(d, i) { + var + x = horizontalQ + ? 0 + : 0 + , + y = verticalQ + ? 0 + : scale(extent[1]) + , + t = 'translate('+x+','+y+')'; + return t + }) + .attr('width', horizontalQ ? objectSize : 0) + .attr('height', verticalQ ? objectSize : 0) + .remove(); + }); - scale.domain(extent).range([Math.min(minObjectSize/2, Math.min(ySize, xSize)/2), Math.min(ySize, xSize)/2]); + // move stuff + spacerFunction(container, ordered, 0); - var ySpacer = groupingSpacer() - .horizontalQ(false) - .moveby('category').numberOfObjects(yDim) - .objectClass(hypenate(objectClass, 'row')) - .objectSize(ySize).spacerSize(ySpacerSize) - .transitionDuration(transitionDuration).easeFunc(easeFunc) - .namespace('row'); - var xSpacer = groupingSpacer() - .horizontalQ(true) - .moveby('category').numberOfObjects(xDim) - .objectClass(objectClass) - .objectSize(xSize).spacerSize(xSpacerSize) - .transitionDuration(transitionDuration).easeFunc(easeFunc); - ySpacer(container, yValues, 0); - container.selectAll('g.'+hypenate(objectClass, 'row')) - .each(function(d, i){ - xSpacer(d3.select(this), xValues, 0); - }); - var cells = container.selectAll('g:not(.to-remove).'+objectClass).data(cellKeys); var parentIndexArray = []; - cells.each(function(d, i){ parentIndexArray.push(Number(d3.select(this).attr('parent-index'))); }); + container.selectAll('g:not(.to-remove).'+objectClass) + .each(function(d, i){parentIndexArray.push(Number(d3.select(this).attr('parent-index')));}); colorFunction$$1 = colorFunction$$1.colorBy() == 'index' ? colorFunction$$1.dataExtent([0, Math.max(...parentIndexArray)]) - : colorFunction$$1.dataExtent([0, Math.max(...vValues)]); + : colorFunction$$1.dataExtent(extent); + - cells.each(function(key, i) { - log('bubbleHeatmap', 'each cell', {key: key, index: i, node: d3.select(this).node()}); + container.selectAll('g.'+objectClass+':not(.to-remove)').each(function(key, i) { + // console.log(key, scale(extent[1]) - scale(valueExtractor(key, i))) var t = d3.select(this), currentData = data[key], - value = vExtractor(key, i), - radius= rExtractor(key, i), + value = valueExtractor(key, i), i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'), fillColor = colorFunction$$1(key, value, i, 'fill'), // prevent duplicate computation strokeColor = colorFunction$$1(key, value, i, 'stroke'); - log('bubbleHeatmap', 'radius',{radius: radius, scaled: scale(radius), extent: extent, range:scale.range()}); - var c = safeSelect(t, 'circle', hypenate(objectClass,'circle')); - c.attr('cx', xSize / 2) - .attr('cy', ySize / 2 ) - .attr('r', scale(radius)) + var bar = utils$1.sel.safeSelect(t, 'rect', 'bar-rect'); + + if (bar.attr('transform') == undefined) { + bar.attr('transform', function(d, i) { + var + x = horizontalQ + ? 0 + : 0 + , + y = verticalQ + ? 0 + : scale(extent[1]) + , + t = 'translate('+x+','+y+')'; + return t + }) + .attr('width', horizontalQ ? objectSize : 0) + .attr('height', verticalQ ? objectSize : 0); + + } + + + bar.transition().duration(transitionDuration).ease(easeFunc) + .attr('transform', function(d, i) { + var + x = horizontalQ + ? objectSize - objectSize * barPercent + : orient == 'right' + ? scale(extent[1]) - scale(value) + : objectSize - objectSize * barPercent + , + y = verticalQ + ? objectSize - objectSize * barPercent + : scale(extent[1]) - scale(value) + , + t = 'translate('+x+','+y+')'; + return t + }) + .attr('width', horizontalQ ? objectSize * barPercent : scale(value)) + .attr('height', verticalQ ? objectSize * barPercent: scale(value)) .attr('fill', fillColor) .attr('stroke', strokeColor) - .attr('stroke-width', bubbleStrokeWidth); + .attr('stroke-width', barStrokeWidth); + + + + t.on('mouseover', function(d, i){ + container.selectAll('g.'+objectClass).style('opacity', 0.2); + t.style('opacity', 1); + bar.attr('stroke-width',barStrokeWidth*2); + }); + t.on('mouseout', function(){ + container.selectAll('g.'+objectClass).style('opacity', 1); + bar.attr('stroke-width', barStrokeWidth); + }); }); - tooltip$$1.selection(cells.selectAll('circle.'+hypenate(objectClass, 'circle'))) + tooltip$$1.selection(container.selectAll('.bar-rect')) .data(data); - // .keys(['r', 'v']) - // .header(function(d, i){return hypenate(data[d][xKey], data[d][yKey]) }) tooltip$$1(); - } - - return bhm; + return bar } /** - * Creates a heatmap + * Creates a bubbleHeatmap * - * {@link https://sumneuron.gitlab.io/d3sm/demos/heatmap-heatmap/index.html Demo} - * @constructor heatmap + * {@link https://sumneuron.gitlab.io/d3sm/demos/bubble-heatmap/index.html Demo} + * @constructor bubbleHeatmap * @param {d3.selection} selection - * @namespace heatmap - * @returns {function} heatmap + * @namespace bubbleHeatmap + * @returns {function} bubbleHeatmap */ -function heatmap( selection ) { +function bubbleHeatmap( selection ) { var /** * Data to plot. Assumed to be a object, where each key corresponds to a cell - * (see {@link heatmap#data}) + * (see {@link bubbleHeatmap#data}) * @param {Object} [data=undefined] - * @memberof heatmap# + * @memberof bubbleHeatmap# * @property */ data, /** - * Amount of horizontal space (in pixels) avaible to render the heatmap in - * (see {@link heatmap#spaceX}) + * Amount of horizontal space (in pixels) avaible to render the bubbleHeatmap in + * (see {@link bubbleHeatmap#spaceX}) * @param {number} [spaceX=undefined] - * @memberof heatmap# + * @memberof bubbleHeatmap# * @property */ spaceX, /** - * Amount of vertical space (in pixels) avaible to render the heatmap in - * (see {@link heatmap.spaceY}) + * Amount of vertical space (in pixels) avaible to render the bubbleHeatmap in + * (see {@link bubbleHeatmap.spaceY}) * @param {number} [spaceY=undefined] - * @memberof heatmap# + * @memberof bubbleHeatmap# * @property */ spaceY, /** * The internal key of the cell specifiying to which x axis key it belongs - * (see {@link heatmap.xKey}) + * (see {@link bubbleHeatmap.xKey}) * @param {string} [xKey='x'] - * @memberof heatmap# + * @memberof bubbleHeatmap# * @property */ xKey = 'x', /** * The internal key of the cell specifiying to which y axis key it belongs - * (see {@link heatmap.yKey}) + * (see {@link bubbleHeatmap.yKey}) * @param {string} [yKey='y'] - * @memberof heatmap# + * @memberof bubbleHeatmap# * @property */ yKey = 'y', - + /** + * The internal key of the cell specifiying what value to use to determine the radius + * (see {@link bubbleHeatmap.rKey}) + * @param {string} [rKey='r'] + * @memberof bubbleHeatmap# + * @property + */ + rKey = 'r', /** * The internal key of the cell specifiying what value to use to determine the color - * (see {@link heatmap.vKey}) + * (see {@link bubbleHeatmap.vKey}) * @param {string} [vKey='v'] - * @memberof heatmap# + * @memberof bubbleHeatmap# * @property */ vKey = 'v', /** * Function for extracting the the value from xKey. - * (see {@link heatmap.xExtractor}) + * (see {@link bubbleHeatmap.xExtractor}) * @param {function} [xExtractor=function(key, i) { return data[key][xKey] }] * @returns {string} - * @memberof heatmap# + * @memberof bubbleHeatmap# * @property */ xExtractor = function(key, i) {return data[key][xKey] }, /** * Function for extracting the the value from yKey. - * (see {@link heatmap.yExtractor}) + * (see {@link bubbleHeatmap.yExtractor}) * @param {function} [yExtractor=function(key, i) { return data[key][yKey] }] * @returns {string} - * @memberof heatmap# + * @memberof bubbleHeatmap# * @property */ yExtractor = function(key, i) { return data[key][yKey] }, - /** - * Function for extracting the the value from vKey. - * (see {@link heatmap.vExtractor}) - * @param {function} [vExtractor=function(key, i) { return data[key][vKey] }] + * Function for extracting the the value from rKey. + * (see {@link bubbleHeatmap.rExtractor}) + * @param {function} [rExtractor=function(key, i) { return data[key][rKey] }] * @returns {number} - * @memberof heatmap# + * @memberof bubbleHeatmap# + * @property + */ + rExtractor = function(key, i) { return data[key][rKey] }, + /** + * Function for extracting the the value from vKey. + * (see {@link bubbleHeatmap.vExtractor}) + * @param {function} [vExtractor=function(key, i) { return data[key][vKey] }] + * @returns {number} + * @memberof bubbleHeatmap# * @property */ vExtractor = function(key, i) { return data[key][vKey] }, /** - * Whether or not to allow heatmap to render elements pass the main spatial dimension - * given the orientation (see {@link heatmap#orient}), where {@link heatmap#orient}="bottom" or {@link heatmap#orient}="top" - * the main dimension is {@link heatmap#spaceX} and where {@link heatmap#orient}="left" or {@link heatmap#orient}="right" - * the main dimension is {@link heatmap#spaceY} + * Whether or not to allow bubbleHeatmap to render elements pass the main spatial dimension + * given the orientation (see {@link bubbleHeatmap#orient}), where {@link bubbleHeatmap#orient}="bottom" or {@link bubbleHeatmap#orient}="top" + * the main dimension is {@link bubbleHeatmap#spaceX} and where {@link bubbleHeatmap#orient}="left" or {@link bubbleHeatmap#orient}="right" + * the main dimension is {@link bubbleHeatmap#spaceY} * @param {boolean} [overflowQ=false] - * @memberof heatmap# + * @memberof bubbleHeatmap# * @property */ overflowQ = false, /** - * Default space for the spacer (percentage) of main dimension given the orientation - * (see {@link heatmap#orient}), where {@link heatmap#orient}="horizontal" - * the main dimension is {@link heatmap#spaceX} and where {@link heatmap#orient}="vertical" - * the main dimension is {@link heatmap#spaceY} between bubbles - * @param {number} [objectSpacer=0.0] - * @memberof heatmap# + * The scale for which the radius values should be transformed by + * @param {d3.scale} [scale=d3.scaleLinear] + * @memberof bubbleHeatmap# * @property */ - objectSpacer = 0.0, + scale = d3.scaleLinear(), /** - * The minimum size that an object can be in the y dimension - * @param {number} [minObjectSize=50] - * @memberof heatmap# + * The padding for the domain of the scale (see {@link bubbleHeatmap#scale}) + * @param {number} [domainPadding=0.5] + * @memberof bubbleHeatmap# * @property */ - yMinObjectSize = 50, + domainPadding = 0.5, + /** - * The minimum size that an object can be in the x dimension - * @param {number} [minObjectSize=50] - * @memberof heatmap# + * Default space for the spacer (percentage) of main dimension given the orientation + * (see {@link bubbleHeatmap#orient}), where {@link bubbleHeatmap#orient}="horizontal" + * the main dimension is {@link bubbleHeatmap#spaceX} and where {@link bubbleHeatmap#orient}="vertical" + * the main dimension is {@link bubbleHeatmap#spaceY} between bubbles + * @param {number} [objectSpacer=0.0] + * @memberof bubbleHeatmap# * @property */ - xMinObjectSize = 50, + objectSpacer = 0.0, /** - * The maximum size that an object can be in the x dimension - * @param {number} [maxObjectSize=100] - * @memberof heatmap# + * The minimum size that an object can be + * @param {number} [minObjectSize=50] + * @memberof bubbleHeatmap# * @property */ - xMaxObjectSize = 100, + minObjectSize = 50, /** - * The maximum size that an object can be in the y dimension + * The maximum size that an object can be * @param {number} [maxObjectSize=100] - * @memberof heatmap# + * @memberof bubbleHeatmap# * @property */ - yMaxObjectSize = 100, + maxObjectSize = 100, /** * The stroke width of the bubbles - * @param {number} [objectStrokeWidth=2] - * @memberof heatmap# + * @param {number} [bubbleStrokeWidth=2] + * @memberof bubbleHeatmap# * @property */ - objectStrokeWidth = 2, + bubbleStrokeWidth = 2, // colorFunc = colorFunction(), /** * Color of the background * @param {string} [backgroundFill="transparent"] - * @memberof heatmap# + * @memberof bubbleHeatmap# * @property */ backgroundFill = 'transparent', /** - * Namespace for all items made by this instance of heatmap - * @param {string} [namespace="d3sm-heatmap"] - * @memberof heatmap# + * Namespace for all items made by this instance of bubbleHeatmap + * @param {string} [namespace="d3sm-bubble"] + * @memberof bubbleHeatmap# * @property */ - namespace = 'd3sm-heatmap', + namespace = 'd3sm-bubble', /** - * Class name for heatmap container ( element) - * @param {string} [objectClass="heatmap"] - * @memberof heatmap# + * Class name for bubble container ( element) + * @param {string} [objectClass="bubble"] + * @memberof bubbleHeatmap# * @property */ - objectClass = 'heatmap', + objectClass = 'bubble', /** * Duration of all transitions of this element * @param {number} [transitionDuration=1000] - * @memberof heatmap# + * @memberof bubbleHeatmap# * @property */ transitionDuration = 1000, /** * Easing function for transitions * @param {d3.ease} [easeFunc=d3.easeExp] - * @memberof heatmap# + * @memberof bubbleHeatmap# * @property */ easeFunc = d3.easeExp, /** * Stores the keys of all the cells - * Calculated after heatmap called. + * Calculated after bubbleHeatmap called. * @param {string[]} [cellKeys=undefined] - * @memberof heatmap# + * @memberof bubbleHeatmap# * @property */ cellKeys, /** - * Stores the list of unique xValues - * Calculated after heatmap called. + * Stores the list of utils.arr.unique xValues + * Calculated after bubbleHeatmap called. * @param {string[]} [xValues=undefined] - * @memberof heatmap# + * @memberof bubbleHeatmap# * @property */ xValues, /** - * Stores the list of unique yValues - * Calculated after heatmap called. + * Stores the list of utils.arr.unique yValues + * Calculated after bubbleHeatmap called. * @param {string[]} [yValues=undefined] - * @memberof heatmap# + * @memberof bubbleHeatmap# * @property */ yValues, /** - * Stores the list of unique vValues - * Calculated after heatmap called. + * Stores the list of utils.arr.unique rValues + * Calculated after bubbleHeatmap called. + * @param {string[]} [rValues=undefined] + * @memberof bubbleHeatmap# + * @property + */ + rValues, + /** + * Stores the list of utils.arr.unique vValues + * Calculated after bubbleHeatmap called. * @param {string[]} [vValues=undefined] - * @memberof heatmap# + * @memberof bubbleHeatmap# * @property */ vValues, - xKeySortingFunction = function(a, b) { return xValues.indexOf(xExtractor(a)) - xValues.indexOf(xExtractor(b)) }, - yKeySortingFunction = function(a, b) { return yValues.indexOf(yExtractor(a)) - yValues.indexOf(yExtractor(b)) }, + xKeySortingFunction = function(a, b) { return xExtractor(a) - xExtractor(b) }, + yKeySortingFunction = function(a, b) { return yExtractor(a) - yExtractor(b) }, /** * Instance of ColorFunction with .colorBy set to 'category' * @function colorFunction - * @memberof heatmap# + * @memberof bubbleHeatmap# * @property */ - colorFunction$$1 = colorFunction().colorBy('category'), + colorFunction$$1 = colorFunction().colorBy('value'), /** * Instance of Tooltip * @function tooltip - * @memberof heatmap# + * @memberof bubbleHeatmap# * @property */ tooltip$$1 = tooltip(), /** - * store the size the heatmap could be in the x dimension - * the actuall size of the heatmap will be the min of xSize and {@link heatmap#ySize} - * Calculated after heatmap called. + * store the size the bubble could be in the x dimension + * the actuall size of the bubble will be the min of xSize and {@link bubbleHeatmap#ySize} + * Calculated after bubbleHeatmap called. * @param {string[]} [xSize=undefined] - * @memberof heatmap# + * @memberof bubbleHeatmap# * @property */ xSize, /** * store the size of the spacer in the x dimension - * Calculated after heatmap called. + * Calculated after bubbleHeatmap called. * @param {string[]} [xSpacerSize=undefined] - * @memberof heatmap# + * @memberof bubbleHeatmap# * @property */ xSpacerSize, /** - * store the size the heatmap could be in the y dimension - * the actuall size of the heatmap will be the min of xSize and {@link heatmap#xSize} - * Calculated after heatmap called. + * store the size the bubble could be in the y dimension + * the actuall size of the bubble will be the min of xSize and {@link bubbleHeatmap#xSize} + * Calculated after bubbleHeatmap called. * @param {string[]} [ySize=undefined] - * @memberof heatmap# + * @memberof bubbleHeatmap# * @property */ ySize, /** * store the size of the spacer in the y dimension. - * Calculated after heatmap called. + * Calculated after bubbleHeatmap called. * @param {string[]} [xSpacerSize=undefined] - * @memberof heatmap# + * @memberof bubbleHeatmap# * @property */ ySpacerSize; @@ -4591,2660 +4437,2586 @@ function heatmap( selection ) { /** * Gets or sets the selection in which items are manipulated * @param {d3.selection} [_=none] - * @returns {heatmap | d3.selection} - * @memberof heatmap + * @returns {bubbleHeatmap | d3.selection} + * @memberof bubbleHeatmap * @property * by default selection = selection */ - hm.selection = function(_) { return arguments.length ? (selection = _, hm) : selection; }; + bhm.selection = function(_) { return arguments.length ? (selection = _, bhm) : selection; }; /** * Gets or sets the data - * (see {@link heatmap#data}) + * (see {@link bubbleHeatmap#data}) * @param {number} [_=none] - * @returns {heatmap | object} - * @memberof heatmap + * @returns {bubbleHeatmap | object} + * @memberof bubbleHeatmap * @property */ - hm.data = function(_) { return arguments.length ? (data = _, hm) : data; }; - // hm.orient = function(_) { return arguments.length ? (orient = _, hm) : orient; } + bhm.data = function(_) { return arguments.length ? (data = _, bhm) : data; }; + // bhm.orient = function(_) { return arguments.length ? (orient = _, bhm) : orient; } /** * Gets or sets the amount of horizontal space in which items are manipulated - * (see {@link heatmap#spaceX}) + * (see {@link bubbleHeatmap#spaceX}) * @param {number} [_=none] should be a number > 0 - * @returns {heatmap | number} - * @memberof heatmap + * @returns {bubbleHeatmap | number} + * @memberof bubbleHeatmap * @property * by default spaceX = undefined */ - hm.spaceX = function(_) { return arguments.length ? (spaceX = _, hm) : spaceX; }; + bhm.spaceX = function(_) { return arguments.length ? (spaceX = _, bhm) : spaceX; }; /** * Gets or sets the amount of vertical space in which items are manipulated - * (see {@link heatmap#spaceY}) + * (see {@link bubbleHeatmap#spaceY}) * @param {number} [_=none] should be a number > 0 - * @returns {heatmap | number} - * @memberof heatmap + * @returns {bubbleHeatmap | number} + * @memberof bubbleHeatmap * @property * by default spaceY = undefined */ - hm.spaceY = function(_) { return arguments.length ? (spaceY = _, hm) : spaceY; }; + bhm.spaceY = function(_) { return arguments.length ? (spaceY = _, bhm) : spaceY; }; /** * Gets or sets the xKey - * (see {@link heatmap#xKey}) + * (see {@link bubbleHeatmap#xKey}) * @param {string} [_=none] - * @returns {heatmap | string} - * @memberof heatmap + * @returns {bubbleHeatmap | string} + * @memberof bubbleHeatmap * @property * by default xKey = 'x' */ - hm.xKey = function(_) { return arguments.length ? (xKey = _, hm) : xKey; }; + bhm.xKey = function(_) { return arguments.length ? (xKey = _, bhm) : xKey; }; /** * Gets or sets the yKey - * (see {@link heatmap#yKey}) + * (see {@link bubbleHeatmap#yKey}) * @param {string} [_=none] - * @returns {heatmap | string} - * @memberof heatmap + * @returns {bubbleHeatmap | string} + * @memberof bubbleHeatmap * @property * by default yKey = 'y' */ - hm.yKey = function(_) { return arguments.length ? (yKey = _, hm) : yKey; }; - + bhm.yKey = function(_) { return arguments.length ? (yKey = _, bhm) : yKey; }; + /** + * Gets or sets the rKey + * (see {@link bubbleHeatmap#rKey}) + * @param {string} [_=none] + * @returns {bubbleHeatmap | string} + * @memberof bubbleHeatmap + * @property + * by default rKey = 'r' + */ + bhm.rKey = function(_) { return arguments.length ? (rKey = _, bhm) : rKey; }; /** * Gets or sets the vKey - * (see {@link heatmap#vKey}) + * (see {@link bubbleHeatmap#vKey}) * @param {string} [_=none] - * @returns {heatmap | string} - * @memberof heatmap + * @returns {bubbleHeatmap | string} + * @memberof bubbleHeatmap * @property * by default vKey = 'y' */ - hm.vKey = function(_) { return arguments.length ? (vKey = _, hm) : vKey; }; + bhm.vKey = function(_) { return arguments.length ? (vKey = _, bhm) : vKey; }; /** * Gets or sets the cellKeys - * (see {@link heatmap#cellKeys}) + * (see {@link bubbleHeatmap#cellKeys}) * @param {string[]} [_=none] - * @returns {heatmap | string[]} - * @memberof heatmap + * @returns {bubbleHeatmap | string[]} + * @memberof bubbleHeatmap * @property * by default cellKeys = undefined */ - hm.cellKeys = function(_) { return arguments.length ? (cellKeys = _, hm) : cellKeys; }; + bhm.cellKeys = function(_) { return arguments.length ? (cellKeys = _, bhm) : cellKeys; }; /** * Gets or sets the xValues - * (see {@link heatmap#xValues}) + * (see {@link bubbleHeatmap#xValues}) * @param {string[]} [_=none] - * @returns {heatmap | string[]} - * @memberof heatmap + * @returns {bubbleHeatmap | string[]} + * @memberof bubbleHeatmap * @property * by default xValues = undefined */ - hm.xValues = function(_) { return arguments.length ? (xValues = _, hm) : xValues; }; + bhm.xValues = function(_) { return arguments.length ? (xValues = _, bhm) : xValues; }; /** * Gets or sets the yValues - * (see {@link heatmap#yValues}) + * (see {@link bubbleHeatmap#yValues}) * @param {string[]} [_=none] - * @returns {heatmap | string[]} - * @memberof heatmap + * @returns {bubbleHeatmap | string[]} + * @memberof bubbleHeatmap * @property * by default yValues = undefined */ - hm.yValues = function(_) { return arguments.length ? (yValues = _, hm) : yValues; }; + bhm.yValues = function(_) { return arguments.length ? (yValues = _, bhm) : yValues; }; /** - * Gets or sets the vValues - * (see {@link heatmap#vValues}) + * Gets or sets the rValues + * (see {@link bubbleHeatmap#rValues}) * @param {number[]} [_=none] - * @returns {heatmap | number[]} - * @memberof heatmap + * @returns {bubbleHeatmap | number[]} + * @memberof bubbleHeatmap * @property - * by default vValues = undefined + * by default rValues = undefined */ - hm.vValues = function(_) { return arguments.length ? (vValues = _, hm) : vValues; }; - - + bhm.rValues = function(_) { return arguments.length ? (rValues = _, bhm) : rValues; }; /** - * Gets or sets the xExtractor - * (see {@link heatmap#xExtractor}) + * Gets or sets the vValues + * (see {@link bubbleHeatmap#vValues}) + * @param {number[]} [_=none] + * @returns {bubbleHeatmap | number[]} + * @memberof bubbleHeatmap + * @property + * by default vValues = undefined + */ + bhm.vValues = function(_) { return arguments.length ? (vValues = _, bhm) : vValues; }; + + + /** + * Gets or sets the xExtractor + * (see {@link bubbleHeatmap#xExtractor}) * @param {function} [_=none] - * @returns {heatmap | function} - * @memberof heatmap + * @returns {bubbleHeatmap | function} + * @memberof bubbleHeatmap * @property * by default xExtractor = undefined */ - hm.xExtractor = function(_) { return arguments.length ? (xExtractor = _, hm) : xExtractor; }; + bhm.xExtractor = function(_) { return arguments.length ? (xExtractor = _, bhm) : xExtractor; }; /** * Gets or sets the yExtractor - * (see {@link heatmap#yExtractor}) + * (see {@link bubbleHeatmap#yExtractor}) * @param {function} [_=none] - * @returns {heatmap | function} - * @memberof heatmap + * @returns {bubbleHeatmap | function} + * @memberof bubbleHeatmap * @property * by default yExtractor = undefined */ - hm.yExtractor = function(_) { return arguments.length ? (yExtractor = _, hm) : yExtractor; }; + bhm.yExtractor = function(_) { return arguments.length ? (yExtractor = _, bhm) : yExtractor; }; + /** + * Gets or sets the rExtractor + * (see {@link bubbleHeatmap#rExtractor}) + * @param {function} [_=none] + * @returns {bubbleHeatmap | function} + * @memberof bubbleHeatmap + * @property + * by default rExtractor = undefined + */ + bhm.rExtractor = function(_) { return arguments.length ? (rExtractor = _, bhm) : rExtractor; }; /** * Gets or sets the vExtractor - * (see {@link heatmap#vExtractor}) + * (see {@link bubbleHeatmap#vExtractor}) * @param {function} [_=none] - * @returns {heatmap | function} - * @memberof heatmap + * @returns {bubbleHeatmap | function} + * @memberof bubbleHeatmap * @property * by default vExtractor = undefined */ - hm.vExtractor = function(_) { return arguments.length ? (vExtractor = _, hm) : vExtractor; }; + bhm.vExtractor = function(_) { return arguments.length ? (vExtractor = _, bhm) : vExtractor; }; /** - * Gets / sets whether or not heatmap is allowed to go beyond specified dimensions - * (see {@link heatmap#spaceX}) + * Gets / sets whether or not bubbleHeatmap is allowed to go beyond specified dimensions + * (see {@link bubbleHeatmap#spaceX}) * @param {boolean} [_=none] - * @returns {heatmap | boolean} - * @memberof heatmap + * @returns {bubbleHeatmap | boolean} + * @memberof bubbleHeatmap * @property * by default overflowQ = false */ - hm.overflowQ = function(_) { return arguments.length ? (overflowQ = _, hm) : overflowQ; }; + bhm.overflowQ = function(_) { return arguments.length ? (overflowQ = _, bhm) : overflowQ; }; /** - * Gets / sets objectSpacer - * (see {@link heatmap#objectSpacer}) - * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * Gets / sets the scale for which the radius of bubbles should be transformed by + * (see {@link bubbleHeatmap#scale}) + * @param {d3.scale} [_=none] + * @returns {bubbleHeatmap | d3.scale} + * @memberof bubbleHeatmap * @property - * by default objectSpacer = 0.0 + * by default scale = d3.scaleLinear() */ - hm.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, objectSpacer) : data; }; + bhm.scale = function(_) { return arguments.length ? (scale = _, bhm) : scale; }; /** - * Gets / sets the minObjectSize - * (see {@link heatmap#minObjectSize}) + * Gets / sets the padding for the domain of the scale + * (see {@link bubbleHeatmap#domainPadding}) * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * @returns {bubbleHeatmap | number} + * @memberof bubbleHeatmap * @property - * by default minObjectSize = 50 + * by default domainPadding = 0.5 */ - hm.yMinObjectSize = function(_) { return arguments.length ? (yMinObjectSize = _, hm) : yMinObjectSize; }; + bhm.domainPadding = function(_) { return arguments.length ? (domainPadding = _, bhm) : domainPadding; }; /** - * Gets / sets the maxObjectSize - * (see {@link heatmap#maxObjectSize}) + * Gets / sets objectSpacer + * (see {@link bubbleHeatmap#objectSpacer}) * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * @returns {bubbleHeatmap | number} + * @memberof bubbleHeatmap * @property - * by default maxObjectSize = 100 + * by default objectSpacer = 0.0 */ - hm.yMaxObjectSize = function(_) { return arguments.length ? (yMaxObjectSize = _, hm) : yMaxObjectSize; }; + bhm.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, objectSpacer) : data; }; /** * Gets / sets the minObjectSize - * (see {@link heatmap#minObjectSize}) + * (see {@link bubbleHeatmap#minObjectSize}) * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * @returns {bubbleHeatmap | number} + * @memberof bubbleHeatmap * @property * by default minObjectSize = 50 */ - hm.xMinObjectSize = function(_) { return arguments.length ? (xMinObjectSize = _, hm) : xMinObjectSize; }; + bhm.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, bhm) : minObjectSize; }; /** * Gets / sets the maxObjectSize - * (see {@link heatmap#maxObjectSize}) + * (see {@link bubbleHeatmap#maxObjectSize}) * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * @returns {bubbleHeatmap | number} + * @memberof bubbleHeatmap * @property * by default maxObjectSize = 100 */ - hm.xMaxObjectSize = function(_) { return arguments.length ? (xMaxObjectSize = _, hm) : xMaxObjectSize; }; + bhm.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, bhm) : maxObjectSize; }; /** - * Gets / sets the objectStrokeWidth - * (see {@link heatmap#objectStrokeWidth}) + * Gets / sets the bubbleStrokeWidth + * (see {@link bubbleHeatmap#bubbleStrokeWidth}) * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * @returns {bubbleHeatmap | number} + * @memberof bubbleHeatmap * @property - * by default objectStrokeWidth = 2 + * by default bubbleStrokeWidth = 2 */ - hm.objectStrokeWidth = function(_) { return arguments.length ? (objectStrokeWidth = _, hm) : objectStrokeWidth; }; + bhm.bubbleStrokeWidth = function(_) { return arguments.length ? (bubbleStrokeWidth = _, bhm) : bubbleStrokeWidth; }; /** * Gets / sets the backgroundFill - * (see {@link heatmap#backgroundFill}) + * (see {@link bubbleHeatmap#backgroundFill}) * @param {string} [_=none] - * @returns {heatmap | string} - * @memberof heatmap + * @returns {bubbleHeatmap | string} + * @memberof bubbleHeatmap * @property * by default backgroundFill = 'transparent' */ - hm.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, hm) : backgroundFill; }; + bhm.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, bhm) : backgroundFill; }; /** * Gets / sets the namespace - * (see {@link heatmap#namespace}) + * (see {@link bubbleHeatmap#namespace}) * @param {string} [_=none] - * @returns {heatmap | string} - * @memberof heatmap + * @returns {bubbleHeatmap | string} + * @memberof bubbleHeatmap * @property - * by default namespace = 'd3sm-heatmap' + * by default namespace = 'd3sm-bubbleHeatmap' */ - hm.namespace = function(_) { return arguments.length ? (namespace = _, hm) : namespace; }; + bhm.namespace = function(_) { return arguments.length ? (namespace = _, bhm) : namespace; }; /** * Gets / sets the objectClass - * (see {@link heatmap#objectClass}) + * (see {@link bubbleHeatmap#objectClass}) * @param {string} [_=none] - * @returns {heatmap | string} - * @memberof heatmap + * @returns {bubbleHeatmap | string} + * @memberof bubbleHeatmap * @property * by default objectClass = 'tick-group' */ - hm.objectClass = function(_) { return arguments.length ? (objectClass = _, hm) : objectClass; }; + bhm.objectClass = function(_) { return arguments.length ? (objectClass = _, bhm) : objectClass; }; /** * Gets / sets the transitionDuration - * (see {@link heatmap#transitionDuration}) + * (see {@link bubbleHeatmap#transitionDuration}) * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * @returns {bubbleHeatmap | number} + * @memberof bubbleHeatmap * @property * by default transitionDuration = 1000 */ - hm.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, hm) : transitionDuration; }; + bhm.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, bhm) : transitionDuration; }; /** * Gets / sets the easeFunc - * (see {@link heatmap#easeFunc}) + * (see {@link bubbleHeatmap#easeFunc}) * @param {d3.ease} [_=none] - * @returns {heatmap | d3.ease} - * @memberof heatmap + * @returns {bubbleHeatmap | d3.ease} + * @memberof bubbleHeatmap * @property * by default easeFunc = d3.easeExp */ - hm.easeFunc = function(_) { return arguments.length ? (easeFunc = _, hm) : easeFunc; }; + bhm.easeFunc = function(_) { return arguments.length ? (easeFunc = _, bhm) : easeFunc; }; /** * Gets / sets the tooltip - * (see {@link heatmap#tooltip}) + * (see {@link bubbleHeatmap#tooltip}) * @param {tooltip} [_=none] - * @returns {heatmap | tooltip} - * @memberof heatmap + * @returns {bubbleHeatmap | tooltip} + * @memberof bubbleHeatmap * @property * by default tooltip = tooltip() */ - hm.tooltip = function(_) { return arguments.length ? (tooltip$$1 = _, hm) : tooltip$$1; }; + bhm.tooltip = function(_) { return arguments.length ? (tooltip$$1 = _, bhm) : tooltip$$1; }; /** * Gets / sets the colorFunction - * (see {@link heatmap#colorFunction}) + * (see {@link bubbleHeatmap#colorFunction}) * @param {colorFunction} [_=none] - * @returns {heatmap | colorFunction} - * @memberof heatmap + * @returns {bubbleHeatmap | colorFunction} + * @memberof bubbleHeatmap * @property * by default colorFunction = colorFunction() */ - hm.colorFunction = function(_) { return arguments.length ? (colorFunction$$1 = _, hm) : colorFunction$$1; }; + bhm.colorFunction = function(_) { return arguments.length ? (colorFunction$$1 = _, bhm) : colorFunction$$1; }; /** * Gets / sets the xSize - * (see {@link heatmap#xSize}) + * (see {@link bubbleHeatmap#xSize}) * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * @returns {bubbleHeatmap | number} + * @memberof bubbleHeatmap * @property * by default xSize = undefined */ - hm.xSize = function(_) { return arguments.length ? (xSize = _, hm) : xSize; }; + bhm.xSize = function(_) { return arguments.length ? (xSize = _, bhm) : xSize; }; /** * Gets / sets the xSpacerSize - * (see {@link heatmap#xSpacerSize}) + * (see {@link bubbleHeatmap#xSpacerSize}) * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * @returns {bubbleHeatmap | number} + * @memberof bubbleHeatmap * @property * by default xSpacerSize = undefined */ - hm.xSpacerSize = function(_) { return arguments.length ? (xSpacerSize = _, hm) : xSpacerSize; }; + bhm.xSpacerSize = function(_) { return arguments.length ? (xSpacerSize = _, bhm) : xSpacerSize; }; /** * Gets / sets the ySize - * (see {@link heatmap#ySize}) + * (see {@link bubbleHeatmap#ySize}) * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * @returns {bubbleHeatmap | number} + * @memberof bubbleHeatmap * @property * by default ySize = undefined */ - hm.ySize = function(_) { return arguments.length ? (ySize = _, hm) : ySize; }; + bhm.ySize = function(_) { return arguments.length ? (ySize = _, bhm) : ySize; }; /** * Gets / sets the ySpacerSize - * (see {@link heatmap#ySpacerSize}) + * (see {@link bubbleHeatmap#ySpacerSize}) * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * @returns {bubbleHeatmap | number} + * @memberof bubbleHeatmap * @property * by default ySpacerSize = undefined */ - hm.ySpacerSize = function(_) { return arguments.length ? (ySpacerSize = _, hm) : ySpacerSize; }; - // hm.yKeySortingFunction = function(_) { return arguments.length ? (yKeySortingFunction = _, hm) : yKeySortingFunction; } - // hm.xKeySortingFunction = function(_) { return arguments.length ? (xKeySortingFunction = _, hm) : xKeySortingFunction; } + bhm.ySpacerSize = function(_) { return arguments.length ? (ySpacerSize = _, bhm) : ySpacerSize; }; + // bhm.yKeySortingFunction = function(_) { return arguments.length ? (yKeySortingFunction = _, bhm) : yKeySortingFunction; } + // bhm.xKeySortingFunction = function(_) { return arguments.length ? (xKeySortingFunction = _, bhm) : xKeySortingFunction; } - function hm() { + function bhm() { var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; - var container = setupContainer( selection, namespace, bgcpRect, backgroundFill ); + var container = utils$1.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); cellKeys = d3.keys(data); - - xValues = unique(cellKeys.map(xExtractor)); - yValues = unique(cellKeys.map(yExtractor)); - vValues = unique(cellKeys.map(vExtractor)); - cellKeys.sort(function(a, b){ return xKeySortingFunction(a, b) || yKeySortingFunction(a, b) }); - log('heatmap', 'cells are sorted by', cellKeys); + utils$1.con.log('bubbleHeatmap', 'cells are sorted by', cellKeys); - log('heatmap', 'x and y keys are', {x: xValues, y:yValues}); + xValues = utils$1.arr.unique(cellKeys.map(xExtractor)); + yValues = utils$1.arr.unique(cellKeys.map(yExtractor)); + rValues = utils$1.arr.unique(cellKeys.map(rExtractor)); + vValues = utils$1.arr.unique(cellKeys.map(vExtractor)); + utils$1.con.log('bubbleHeatmap', 'x and y keys are', {x: xValues, y:yValues}); var xDim = xValues.length, yDim = yValues.length; - ySize = calculateWidthOfObject(spaceY, yDim, yMinObjectSize, yMaxObjectSize, objectSpacer, overflowQ); - xSize = calculateWidthOfObject(spaceX, xDim, xMinObjectSize, xMaxObjectSize, objectSpacer, overflowQ); - ySpacerSize = calculateWidthOfSpacer(yValues, spaceY, ySize, yDim, objectSpacer, overflowQ); - xSpacerSize = calculateWidthOfSpacer(xValues, spaceX, xSize, xDim, objectSpacer, overflowQ); - // console.table({ - // x:{ - // object: xSize, - // spacer: xSpacerSize, - // dim: xDim - // }, - // y:{ - // object: ySize, - // spacer: ySpacerSize, - // dim: yDim - // } - // - // }) - log('heatmap', 'size of', {x: xSize, y: ySize}); + var extent = [Math.min(...rValues) - domainPadding,Math.max(...rValues) + domainPadding]; + + ySize = utils$1.math.calculateWidthOfObject(spaceY, yDim, minObjectSize, maxObjectSize, objectSpacer, overflowQ); + xSize = utils$1.math.calculateWidthOfObject(spaceX, xDim, minObjectSize, maxObjectSize, objectSpacer, overflowQ); + ySpacerSize = utils$1.math.calculateWidthOfSpacer(yValues, spaceY, ySize, yDim, objectSpacer, overflowQ); + xSpacerSize = utils$1.math.calculateWidthOfSpacer(xValues, spaceX, xSize, xDim, objectSpacer, overflowQ); + utils$1.con.log('bubbleHeatmap', 'size of', {x: xSize, y: ySize}); + scale.domain(extent).range([Math.min(minObjectSize/2, Math.min(ySize, xSize)/2), Math.min(ySize, xSize)/2]); + var ySpacer = groupingSpacer() .horizontalQ(false) - .moveby('category') - .numberOfObjects(yDim) - .objectClass(hypenate(objectClass, 'row')) - .objectSize(ySize + ySpacerSize) - .spacerSize(0) - .transitionDuration(transitionDuration) - .easeFunc(easeFunc) + .moveby('category').numberOfObjects(yDim) + .objectClass(utils$1.str.hypenate(objectClass, 'row')) + .objectSize(ySize).spacerSize(ySpacerSize) + .transitionDuration(transitionDuration).easeFunc(easeFunc) .namespace('row'); var xSpacer = groupingSpacer() .horizontalQ(true) - .moveby('category') - .numberOfObjects(xDim) + .moveby('category').numberOfObjects(xDim) .objectClass(objectClass) - .objectSize(xSize + xSpacerSize) - .spacerSize(0) - .transitionDuration(transitionDuration) - .easeFunc(easeFunc); + .objectSize(xSize).spacerSize(xSpacerSize) + .transitionDuration(transitionDuration).easeFunc(easeFunc); ySpacer(container, yValues, 0); - container.selectAll('g.'+hypenate(objectClass, 'row')) - .each(function(d, i){ xSpacer(d3.select(this), xValues, 0); }); - - var cells = container.selectAll('g:not(.to-remove).'+objectClass); - - - if (cellKeys.length != yValues.length * xValues.length) { - var lookup = {}; - cellKeys.map(function(k, i){ - lookup[xExtractor(k)+'::'+yExtractor(k)] = k; - }); - - var positionedCellKeys = []; - for (var i = 0; i < yValues.length; i++) { - for (var j = 0; j < xValues.length; j++) { - var lookupValue = lookup[xValues[j]+"::"+yValues[i]]; - if (lookupValue == undefined) { - positionedCellKeys.push(undefined); - } else { - positionedCellKeys.push(lookupValue); - } - } - } - - cells.data(positionedCellKeys); - - // maybe breaks this - // !!!!!! IMPORTANT NOTE TODO LOOK HERE BUG - cellKeys = positionedCellKeys; - } else { - cells.data(cellKeys); - } - - + container.selectAll('g.'+utils$1.str.hypenate(objectClass, 'row')) + .each(function(d, i){ + xSpacer(d3.select(this), xValues, 0); + }); + var cells = container.selectAll('g:not(.to-remove).'+objectClass).data(cellKeys); var parentIndexArray = []; cells.each(function(d, i){ parentIndexArray.push(Number(d3.select(this).attr('parent-index'))); }); + colorFunction$$1 = colorFunction$$1.colorBy() == 'index' ? colorFunction$$1.dataExtent([0, Math.max(...parentIndexArray)]) : colorFunction$$1.dataExtent([0, Math.max(...vValues)]); cells.each(function(key, i) { - log('heatmap', 'each cell', {key: key, index: i, node: d3.select(this).node()}); + utils$1.con.log('bubbleHeatmap', 'each cell', {key: key, index: i, node: d3.select(this).node()}); - var t = d3.select(this); - if (key == undefined) {return} - var currentData = data[key], + var t = d3.select(this), + currentData = data[key], value = vExtractor(key, i), + radius= rExtractor(key, i), i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'), fillColor = colorFunction$$1(key, value, i, 'fill'), // prevent duplicate computation strokeColor = colorFunction$$1(key, value, i, 'stroke'); - var c = safeSelect(t, 'rect', hypenate(objectClass,'rect')); - c.attr('width', xSize + xSpacerSize - objectStrokeWidth) - .attr('height', ySize + ySpacerSize - objectStrokeWidth) + utils$1.con.log('bubbleHeatmap', 'radius',{radius: radius, scaled: scale(radius), extent: extent, range:scale.range()}); + + var c = utils$1.sel.safeSelect(t, 'circle', utils$1.str.hypenate(objectClass,'circle')); + c.attr('cx', xSize / 2) + .attr('cy', ySize / 2 ) + .attr('r', scale(radius)) .attr('fill', fillColor) - .attr('x', objectStrokeWidth/2) - .attr('y', objectStrokeWidth/2) - .attr('stroke', "#000") - .attr('stroke-width', objectStrokeWidth); + .attr('stroke', strokeColor) + .attr('stroke-width', bubbleStrokeWidth); }); - tooltip$$1.selection(cells.selectAll('rect.'+hypenate(objectClass, 'rect'))) + tooltip$$1.selection(cells.selectAll('circle.'+utils$1.str.hypenate(objectClass, 'circle'))) .data(data); // .keys(['r', 'v']) - // .header(function(d, i){return hypenate(data[d][xKey], data[d][yKey]) }) + // .header(function(d, i){return utils.str.hypenate(data[d][xKey], data[d][yKey]) }) tooltip$$1(); } - return hm; + return bhm; } -/******************************************************************************* -** ** -** ** -** BOX AND WHISKER ** -** ** -** ** -*******************************************************************************/ /** - * Creates a boxwhisker + * Creates a heatmap * - * {@link https://sumneuron.gitlab.io/d3sm/demos/box-whiskers/index.html Demo} - * @constructor boxwhisker + * {@link https://sumneuron.gitlab.io/d3sm/demos/heatmap-heatmap/index.html Demo} + * @constructor heatmap * @param {d3.selection} selection - * @namespace boxwhisker - * @returns {function} boxwhisker + * @namespace heatmap + * @returns {function} heatmap */ -function boxwhisker( selection ) { - var +function heatmap( selection ) { + var /** - * Data to plot. Assumed to be a object, where each key corresponds to a box - * (see {@link boxwhisker#data}) + * Data to plot. Assumed to be a object, where each key corresponds to a cell + * (see {@link heatmap#data}) * @param {Object} [data=undefined] - * @memberof boxwhisker# + * @memberof heatmap# * @property */ data, + /** - * Which direction to render the boxes in - * (see {@link boxwhisker#orient}) - * @param {number} [orient='horizontal'] - * @memberof boxwhisker# - * @property - */ - orient = 'horizontal', - /** - * Amount of horizontal space (in pixels) avaible to render the boxwhisker in - * (see {@link boxwhisker#spaceX}) + * Amount of horizontal space (in pixels) avaible to render the heatmap in + * (see {@link heatmap#spaceX}) * @param {number} [spaceX=undefined] - * @memberof boxwhisker# + * @memberof heatmap# * @property */ spaceX, /** - * Amount of vertical space (in pixels) avaible to render the boxwhisker in - * (see {@link boxwhisker.spaceY}) + * Amount of vertical space (in pixels) avaible to render the heatmap in + * (see {@link heatmap.spaceY}) * @param {number} [spaceY=undefined] - * @memberof boxwhisker# + * @memberof heatmap# * @property */ spaceY, + /** - * Whether or not to allow boxwhisker to render elements pass the main spatial dimension - * given the orientation (see {@link boxwhisker#orient}), where {@link boxwhisker#orient}="horizontal" - * the main dimension is {@link boxwhisker#spaceX} and where {@link boxwhisker#orient}="vertical" - * the main dimension is {@link boxwhisker#spaceY} - * @param {boolean} [overflowQ=false] - * @memberof boxwhisker# + * The internal key of the cell specifiying to which x axis key it belongs + * (see {@link heatmap.xKey}) + * @param {string} [xKey='x'] + * @memberof heatmap# * @property */ - overflowQ = true, - + xKey = 'x', /** - * An array - putatively of other arrays - depicting how boxes should be arranged - * @param {Array[]} [grouping=undefined] - * @memberof boxwhisker# + * The internal key of the cell specifiying to which y axis key it belongs + * (see {@link heatmap.yKey}) + * @param {string} [yKey='y'] + * @memberof heatmap# * @property */ - grouping, - quartilesKey = 'quartiles', // key in object where quartiles are stored - quartilesKeys = ["0.00", "0.25", "0.50", "0.75", "1.00"], // keys in quartiles object mapping values to min, q1, q2, q3 and max - + yKey = 'y', /** - * How to get the value of the boxwhisker - * @param {function} [valueExtractor=function(key, index) { return data[key][quartilesKey] }] - * @memberof boxwhisker# + * The internal key of the cell specifiying what value to use to determine the color + * (see {@link heatmap.vKey}) + * @param {string} [vKey='v'] + * @memberof heatmap# * @property */ - valueExtractor = function(key, index) { return data[key][quartilesKey] }, + vKey = 'v', + /** - * How to sort the boxes - if {@link boxwhisker#grouping} is not provided. - * @param {function} [sortingFunction=descending] - * @memberof boxwhisker# + * Function for extracting the the value from xKey. + * (see {@link heatmap.xExtractor}) + * @param {function} [xExtractor=function(key, i) { return data[key][xKey] }] + * @returns {string} + * @memberof heatmap# * @property - * default - * function(keyA, keyB) {return d3.descending( - * valueExtractor(keyA)[quartilesKeys[4]], - * valueExtractor(keyB)[quartilesKeys[4]] - * )} */ - sortingFunction = function(keyA, keyB) {return d3.descending( - valueExtractor(keyA)[quartilesKeys[4]], - valueExtractor(keyB)[quartilesKeys[4]] - )}, + xExtractor = function(key, i) {return data[key][xKey] }, /** - * The scale for which boxwhisker values should be transformed by - * @param {d3.scale} [scale=d3.scaleLinear] - * @memberof boxwhisker# + * Function for extracting the the value from yKey. + * (see {@link heatmap.yExtractor}) + * @param {function} [yExtractor=function(key, i) { return data[key][yKey] }] + * @returns {string} + * @memberof heatmap# * @property */ - scale = d3.scaleLinear(), + yExtractor = function(key, i) { return data[key][yKey] }, + /** - * The padding for the domain of the scale (see {@link boxwhisker#scale}) - * @param {number} [domainPadding=0.5] - * @memberof boxwhisker# + * Function for extracting the the value from vKey. + * (see {@link heatmap.vExtractor}) + * @param {function} [vExtractor=function(key, i) { return data[key][vKey] }] + * @returns {number} + * @memberof heatmap# * @property */ - domainPadding = 0.5, + vExtractor = function(key, i) { return data[key][vKey] }, + + /** - * Default space for the spacer (percentage) of main dimension given the orientation - * (see {@link boxwhisker#orient}), where {@link boxwhisker#orient}="horizontal" - * the main dimension is {@link boxwhisker#spaceX} and where {@link boxwhisker#orient}="vertical" - * the main dimension is {@link boxwhisker#spaceY} between boxes - * @param {number} [objectSpacer=0.05] - * @memberof boxwhisker# + * Whether or not to allow heatmap to render elements pass the main spatial dimension + * given the orientation (see {@link heatmap#orient}), where {@link heatmap#orient}="bottom" or {@link heatmap#orient}="top" + * the main dimension is {@link heatmap#spaceX} and where {@link heatmap#orient}="left" or {@link heatmap#orient}="right" + * the main dimension is {@link heatmap#spaceY} + * @param {boolean} [overflowQ=false] + * @memberof heatmap# * @property */ - objectSpacer = 0.05, + overflowQ = false, + /** - * The minimum size that an object can be - * @param {number} [minObjectSize=15] - * @memberof boxwhisker# + * Default space for the spacer (percentage) of main dimension given the orientation + * (see {@link heatmap#orient}), where {@link heatmap#orient}="horizontal" + * the main dimension is {@link heatmap#spaceX} and where {@link heatmap#orient}="vertical" + * the main dimension is {@link heatmap#spaceY} between bubbles + * @param {number} [objectSpacer=0.0] + * @memberof heatmap# * @property */ - minObjectSize = 15, + objectSpacer = 0.0, /** - * The maximum size that an object can be - * @param {number} [maxObjectSize=50] - * @memberof boxwhisker# + * The minimum size that an object can be in the y dimension + * @param {number} [minObjectSize=50] + * @memberof heatmap# * @property */ - maxObjectSize = 50, + yMinObjectSize = 50, /** - * Percent of box and whisker size that whiskers will be rendered - * see {@link boxwhisker#objectSize} - * @param {number} [whiskerWidthPercent=0.6] - * @memberof boxwhisker# + * The minimum size that an object can be in the x dimension + * @param {number} [minObjectSize=50] + * @memberof heatmap# * @property */ - whiskerWidthPercent = .6, + xMinObjectSize = 50, /** - * Instance of ColorFunction - * @param {function} [colorFunction = colorFunction()] - * @memberof boxwhisker# + * The maximum size that an object can be in the x dimension + * @param {number} [maxObjectSize=100] + * @memberof heatmap# * @property */ - colorFunction$$1 = colorFunction(), + xMaxObjectSize = 100, /** - * The stroke width of the boxes - * @param {number} [boxStrokeWidth=2] - * @memberof boxwhisker# + * The maximum size that an object can be in the y dimension + * @param {number} [maxObjectSize=100] + * @memberof heatmap# * @property */ - boxStrokeWidth = 2, + yMaxObjectSize = 100, + + /** - * The stroke width of the whiskers - * @param {number} [whiskerStrokeWidth=2] - * @memberof boxwhisker# + * The stroke width of the bubbles + * @param {number} [objectStrokeWidth=2] + * @memberof heatmap# * @property */ - whiskerStrokeWidth = 2, + objectStrokeWidth = 2, + // colorFunc = colorFunction(), /** * Color of the background * @param {string} [backgroundFill="transparent"] - * @memberof boxwhisker# + * @memberof heatmap# * @property */ backgroundFill = 'transparent', /** - * Namespace for all items made by this instance of boxwhisker - * @param {string} [namespace="d3sm-box-whisker"] - * @memberof boxwhisker# + * Namespace for all items made by this instance of heatmap + * @param {string} [namespace="d3sm-heatmap"] + * @memberof heatmap# * @property */ - namespace = 'd3sm-box-whisker', + namespace = 'd3sm-heatmap', /** - * Class name for boxwhisker container ( element) - * @param {string} [objectClass="box-whisk"] - * @memberof boxwhisker# + * Class name for heatmap container ( element) + * @param {string} [objectClass="heatmap"] + * @memberof heatmap# * @property */ - objectClass = 'box-whisk', - + objectClass = 'heatmap', /** * Duration of all transitions of this element * @param {number} [transitionDuration=1000] - * @memberof boxwhisker# + * @memberof heatmap# * @property */ transitionDuration = 1000, /** * Easing function for transitions * @param {d3.ease} [easeFunc=d3.easeExp] - * @memberof boxwhisker# + * @memberof heatmap# * @property */ easeFunc = d3.easeExp, /** - * The keys of the boxes - * @param {string[]} [boxKeys=undefined] - * @memberof boxwhisker# + * Stores the keys of all the cells + * Calculated after heatmap called. + * @param {string[]} [cellKeys=undefined] + * @memberof heatmap# * @property */ - boxKeys, + cellKeys, /** - * The values of the boxes - * @param {string[]} [boxValues=undefined] - * @memberof boxwhisker# + * Stores the list of utils.arr.unique xValues + * Calculated after heatmap called. + * @param {string[]} [xValues=undefined] + * @memberof heatmap# * @property */ - boxValues, + xValues, /** - * The objectSize (actual width) used by the boxes - * @param {number} [objectSize=undefined] - * @memberof boxwhisker# + * Stores the list of utils.arr.unique yValues + * Calculated after heatmap called. + * @param {string[]} [yValues=undefined] + * @memberof heatmap# * @property */ - objectSize, + yValues, /** - * The spacerSize (actual width) used by the spacers between the boxes - * @param {number} [spacerSize=undefined] - * @memberof boxwhisker# + * Stores the list of utils.arr.unique vValues + * Calculated after heatmap called. + * @param {string[]} [vValues=undefined] + * @memberof heatmap# * @property */ - spacerSize, + vValues, + + xKeySortingFunction = function(a, b) { return xValues.indexOf(xExtractor(a)) - xValues.indexOf(xExtractor(b)) }, + yKeySortingFunction = function(a, b) { return yValues.indexOf(yExtractor(a)) - yValues.indexOf(yExtractor(b)) }, /** - * Instance of Tooltip - * @param {function} [tooltip=tooltip()] - * @memberof boxwhisker# + * Instance of ColorFunction with .colorBy set to 'category' + * @function colorFunction + * @memberof heatmap# * @property */ - tooltip$$1 = tooltip(); - /** - * Gets or sets the selection in which items are manipulated - * @param {d3.selection} [_=none] - * @returns {boxwhisker | d3.selection} - * @memberof boxwhisker - * @property - * by default selection = selection - */ - boxwhisker.selection = function(_) { return arguments.length ? (selection = _, boxwhisker) : selection; }; + colorFunction$$1 = colorFunction().colorBy('category'), /** - * Gets or sets the data - * (see {@link boxwhisker#data}) - * @param {number} [_=none] - * @returns {boxwhisker | object} - * @memberof boxwhisker - * @property - */ - boxwhisker.data = function(_) { return arguments.length ? (data = _, boxwhisker) : data; }; + * Instance of Tooltip + * @function tooltip + * @memberof heatmap# + * @property + */ + tooltip$$1 = tooltip(), + /** - * Gets or sets the orient of the boxes - * (see {@link boxwhisker#orient}) - * @param {number} [_=none] - * @returns {boxwhisker | object} - * @memberof boxwhisker - * @property - */ - boxwhisker.orient = function(_) { return arguments.length ? (orient = _, boxwhisker) : orient; }; + * store the size the heatmap could be in the x dimension + * the actuall size of the heatmap will be the min of xSize and {@link heatmap#ySize} + * Calculated after heatmap called. + * @param {string[]} [xSize=undefined] + * @memberof heatmap# + * @property + */ + xSize, /** - * Gets or sets the amount of horizontal space in which items are manipulated - * (see {@link boxwhisker#spaceX}) + * store the size of the spacer in the x dimension + * Calculated after heatmap called. + * @param {string[]} [xSpacerSize=undefined] + * @memberof heatmap# + * @property + */ + xSpacerSize, + + /** + * store the size the heatmap could be in the y dimension + * the actuall size of the heatmap will be the min of xSize and {@link heatmap#xSize} + * Calculated after heatmap called. + * @param {string[]} [ySize=undefined] + * @memberof heatmap# + * @property + */ + ySize, + /** + * store the size of the spacer in the y dimension. + * Calculated after heatmap called. + * @param {string[]} [xSpacerSize=undefined] + * @memberof heatmap# + * @property + */ + ySpacerSize; + + /** + * Gets or sets the selection in which items are manipulated + * @param {d3.selection} [_=none] + * @returns {heatmap | d3.selection} + * @memberof heatmap + * @property + * by default selection = selection + */ + hm.selection = function(_) { return arguments.length ? (selection = _, hm) : selection; }; + /** + * Gets or sets the data + * (see {@link heatmap#data}) + * @param {number} [_=none] + * @returns {heatmap | object} + * @memberof heatmap + * @property + */ + hm.data = function(_) { return arguments.length ? (data = _, hm) : data; }; + // hm.orient = function(_) { return arguments.length ? (orient = _, hm) : orient; } + /** + * Gets or sets the amount of horizontal space in which items are manipulated + * (see {@link heatmap#spaceX}) * @param {number} [_=none] should be a number > 0 - * @returns {boxwhisker | number} - * @memberof boxwhisker + * @returns {heatmap | number} + * @memberof heatmap * @property * by default spaceX = undefined */ - boxwhisker.spaceX = function(_) { return arguments.length ? (spaceX = _, boxwhisker) : spaceX; }; + hm.spaceX = function(_) { return arguments.length ? (spaceX = _, hm) : spaceX; }; /** * Gets or sets the amount of vertical space in which items are manipulated - * (see {@link boxwhisker#spaceY}) + * (see {@link heatmap#spaceY}) * @param {number} [_=none] should be a number > 0 - * @returns {boxwhisker | number} - * @memberof boxwhisker + * @returns {heatmap | number} + * @memberof heatmap * @property * by default spaceY = undefined */ - boxwhisker.spaceY = function(_) { return arguments.length ? (spaceY = _, boxwhisker) : spaceY; }; + hm.spaceY = function(_) { return arguments.length ? (spaceY = _, hm) : spaceY; }; + /** - * Gets / sets whether or not boxwhisker is allowed to go beyond specified dimensions - * (see {@link boxwhisker#overflowQ}) - * @param {boolean} [_=none] - * @returns {boxwhisker | boolean} - * @memberof boxwhisker + * Gets or sets the xKey + * (see {@link heatmap#xKey}) + * @param {string} [_=none] + * @returns {heatmap | string} + * @memberof heatmap * @property - * by default overflowQ = false + * by default xKey = 'x' */ - boxwhisker.overflowQ = function(_) { return arguments.length ? (overflowQ = _, boxwhisker) : overflowQ; }; + hm.xKey = function(_) { return arguments.length ? (xKey = _, hm) : xKey; }; /** - * Gets / sets the grouping of the boxes - * (see {@link boxwhisker#grouping}) - * @param {Array[]} [_=none] - * @returns {boxwhisker | Array[]} - * @memberof boxwhisker + * Gets or sets the yKey + * (see {@link heatmap#yKey}) + * @param {string} [_=none] + * @returns {heatmap | string} + * @memberof heatmap * @property - * by default grouping = undefined + * by default yKey = 'y' */ - boxwhisker.grouping = function(_) { return arguments.length ? (grouping = _, boxwhisker) : grouping; }; + hm.yKey = function(_) { return arguments.length ? (yKey = _, hm) : yKey; }; + /** - * Gets / sets the quartilesKey - * (see {@link boxwhisker#quartilesKey}) + * Gets or sets the vKey + * (see {@link heatmap#vKey}) * @param {string} [_=none] - * @returns {boxwhisker | string} - * @memberof boxwhisker + * @returns {heatmap | string} + * @memberof heatmap * @property - * by default quartilesKey = quartiles + * by default vKey = 'y' */ - boxwhisker.quartilesKey = function(_) { return arguments.length ? (quartilesKey = _, boxwhisker) : quartilesKey; }; + hm.vKey = function(_) { return arguments.length ? (vKey = _, hm) : vKey; }; + /** - * Gets / sets the quartilesKeys - * (see {@link boxwhisker#quartilesKeys}) + * Gets or sets the cellKeys + * (see {@link heatmap#cellKeys}) * @param {string[]} [_=none] - * @returns {boxwhisker | string[]} - * @memberof boxwhisker + * @returns {heatmap | string[]} + * @memberof heatmap * @property - * by default quartilesKeys = [Q0, Q1, Q2, Q3, Q4] + * by default cellKeys = undefined */ - boxwhisker.quartilesKeys = function(_) { return arguments.length ? (quartilesKeys = _, boxwhisker) : quartilesKeys; }; + hm.cellKeys = function(_) { return arguments.length ? (cellKeys = _, hm) : cellKeys; }; /** - * Gets / sets the valueExtractor - * (see {@link boxwhisker#valueExtractor}) - * @param {function} [_=none] - * @returns {boxwhisker | function} - * @memberof boxwhisker + * Gets or sets the xValues + * (see {@link heatmap#xValues}) + * @param {string[]} [_=none] + * @returns {heatmap | string[]} + * @memberof heatmap * @property - * by default valueExtractor = function(key, index) { return data[key][quartilesKey] }, + * by default xValues = undefined + */ + hm.xValues = function(_) { return arguments.length ? (xValues = _, hm) : xValues; }; + /** + * Gets or sets the yValues + * (see {@link heatmap#yValues}) + * @param {string[]} [_=none] + * @returns {heatmap | string[]} + * @memberof heatmap + * @property + * by default yValues = undefined + */ + hm.yValues = function(_) { return arguments.length ? (yValues = _, hm) : yValues; }; + /** + * Gets or sets the vValues + * (see {@link heatmap#vValues}) + * @param {number[]} [_=none] + * @returns {heatmap | number[]} + * @memberof heatmap + * @property + * by default vValues = undefined */ + hm.vValues = function(_) { return arguments.length ? (vValues = _, hm) : vValues; }; + - boxwhisker.valueExtractor = function(_) { return arguments.length ? (valueExtractor = _, boxwhisker) : valueExtractor; }; /** - * Gets / sets the sortingFunction - * (see {@link boxwhisker#sortingFunction}) + * Gets or sets the xExtractor + * (see {@link heatmap#xExtractor}) * @param {function} [_=none] - * @returns {boxwhisker | function} - * @memberof boxwhisker + * @returns {heatmap | function} + * @memberof heatmap * @property - * by default sortingFunction = function(keyA, keyB) {return d3.descending( - * valueExtractor(keyA)[quartilesKeys[4]], - * valueExtractor(keyB)[quartilesKeys[4]] - * )}, + * by default xExtractor = undefined */ - boxwhisker.sortingFunction = function(_) { return arguments.length ? (sortingFunction = _, boxwhisker) : sortingFunction; }; + hm.xExtractor = function(_) { return arguments.length ? (xExtractor = _, hm) : xExtractor; }; /** - * Gets / sets the scale for which the boxwhisker values should be transformed by - * (see {@link boxwhisker#scale}) - * @param {d3.scale} [_=none] - * @returns {boxwhisker | d3.scale} - * @memberof boxwhisker + * Gets or sets the yExtractor + * (see {@link heatmap#yExtractor}) + * @param {function} [_=none] + * @returns {heatmap | function} + * @memberof heatmap * @property - * by default scale = d3.scaleLinear() + * by default yExtractor = undefined */ - boxwhisker.scale = function(_) { return arguments.length ? (scale = _, boxwhisker) : scale; }; + hm.yExtractor = function(_) { return arguments.length ? (yExtractor = _, hm) : yExtractor; }; /** - * Gets / sets the padding for the domain of the scale - * (see {@link boxwhisker#domainPadding}) - * @param {number} [_=none] - * @returns {boxwhisker | number} - * @memberof boxwhisker + * Gets or sets the vExtractor + * (see {@link heatmap#vExtractor}) + * @param {function} [_=none] + * @returns {heatmap | function} + * @memberof heatmap * @property - * by default domainPadding = 0.5 + * by default vExtractor = undefined */ - boxwhisker.domainPadding = function(_) { return arguments.length ? (domainPadding = _, boxwhisker) : domainPadding; }; + hm.vExtractor = function(_) { return arguments.length ? (vExtractor = _, hm) : vExtractor; }; + + /** + * Gets / sets whether or not heatmap is allowed to go beyond specified dimensions + * (see {@link heatmap#spaceX}) + * @param {boolean} [_=none] + * @returns {heatmap | boolean} + * @memberof heatmap + * @property + * by default overflowQ = false + */ + hm.overflowQ = function(_) { return arguments.length ? (overflowQ = _, hm) : overflowQ; }; /** * Gets / sets objectSpacer - * (see {@link boxwhisker#objectSpacer}) + * (see {@link heatmap#objectSpacer}) * @param {number} [_=none] - * @returns {boxwhisker | number} - * @memberof boxwhisker + * @returns {heatmap | number} + * @memberof heatmap * @property - * by default objectSpacer = 0.05 + * by default objectSpacer = 0.0 */ - boxwhisker.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, boxwhisker) : objectSpacer; }; + hm.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, objectSpacer) : data; }; /** * Gets / sets the minObjectSize - * (see {@link boxwhisker#minObjectSize}) + * (see {@link heatmap#minObjectSize}) * @param {number} [_=none] - * @returns {boxwhisker | number} - * @memberof boxwhisker + * @returns {heatmap | number} + * @memberof heatmap * @property - * by default minObjectSize = 15 + * by default minObjectSize = 50 */ - boxwhisker.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, boxwhisker) : minObjectSize; }; + hm.yMinObjectSize = function(_) { return arguments.length ? (yMinObjectSize = _, hm) : yMinObjectSize; }; /** * Gets / sets the maxObjectSize - * (see {@link boxwhisker#maxObjectSize}) + * (see {@link heatmap#maxObjectSize}) * @param {number} [_=none] - * @returns {boxwhisker | number} - * @memberof boxwhisker + * @returns {heatmap | number} + * @memberof heatmap * @property - * by default maxObjectSize = 50 + * by default maxObjectSize = 100 */ - boxwhisker.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, boxwhisker) : maxObjectSize; }; + hm.yMaxObjectSize = function(_) { return arguments.length ? (yMaxObjectSize = _, hm) : yMaxObjectSize; }; /** - * Gets / sets the whiskerWidthPercent - * (see {@link boxwhisker#whiskerWidthPercent}) + * Gets / sets the minObjectSize + * (see {@link heatmap#minObjectSize}) * @param {number} [_=none] - * @returns {boxwhisker | number} - * @memberof boxwhisker - * @property - * by default maxObjectSize = 0.6 - */ - boxwhisker.whiskerWidthPercent = function(_) { return arguments.length ? (whiskerWidthPercent = _, boxwhisker) : whiskerWidthPercent; }; - /** - * Gets / sets the colorFunction - * (see {@link boxwhisker#colorFunction}) - * @param {colorFunction} [_=none] - * @returns {boxwhisker | colorFunction} - * @memberof boxwhisker + * @returns {heatmap | number} + * @memberof heatmap * @property - * by default colorFunction = colorFunction() + * by default minObjectSize = 50 */ - boxwhisker.colorFunction = function(_) { return arguments.length ? (colorFunction$$1 = _, boxwhisker) : colorFunction$$1; }; + hm.xMinObjectSize = function(_) { return arguments.length ? (xMinObjectSize = _, hm) : xMinObjectSize; }; /** - * Gets / sets the boxStrokeWidth - * (see {@link boxwhisker#boxStrokeWidth}) + * Gets / sets the maxObjectSize + * (see {@link heatmap#maxObjectSize}) * @param {number} [_=none] - * @returns {boxwhisker | number} - * @memberof boxwhisker + * @returns {heatmap | number} + * @memberof heatmap * @property - * by default boxStrokeWidth = 2 + * by default maxObjectSize = 100 */ - boxwhisker.boxStrokeWidth = function(_) { return arguments.length ? (boxStrokeWidth = _, boxwhisker) : boxStrokeWidth; }; + hm.xMaxObjectSize = function(_) { return arguments.length ? (xMaxObjectSize = _, hm) : xMaxObjectSize; }; /** - * Gets / sets the whiskerStrokeWidth - * (see {@link boxwhisker#whiskerStrokeWidth}) + * Gets / sets the objectStrokeWidth + * (see {@link heatmap#objectStrokeWidth}) * @param {number} [_=none] - * @returns {boxwhisker | number} - * @memberof boxwhisker + * @returns {heatmap | number} + * @memberof heatmap * @property - * by default whiskerStrokeWidth = 2 + * by default objectStrokeWidth = 2 */ - boxwhisker.whiskerStrokeWidth = function(_) { return arguments.length ? (whiskerStrokeWidth = _, boxwhisker) : whiskerStrokeWidth; }; - + hm.objectStrokeWidth = function(_) { return arguments.length ? (objectStrokeWidth = _, hm) : objectStrokeWidth; }; /** * Gets / sets the backgroundFill - * (see {@link boxwhisker#backgroundFill}) + * (see {@link heatmap#backgroundFill}) * @param {string} [_=none] - * @returns {boxwhisker | string} - * @memberof boxwhisker + * @returns {heatmap | string} + * @memberof heatmap * @property * by default backgroundFill = 'transparent' */ - boxwhisker.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, boxwhisker) : backgroundFill; }; + hm.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, hm) : backgroundFill; }; /** * Gets / sets the namespace - * (see {@link boxwhisker#namespace}) + * (see {@link heatmap#namespace}) * @param {string} [_=none] - * @returns {boxwhisker | string} - * @memberof boxwhisker + * @returns {heatmap | string} + * @memberof heatmap * @property - * by default namespace = 'd3sm-boxwhisker' + * by default namespace = 'd3sm-heatmap' */ - boxwhisker.namespace = function(_) { return arguments.length ? (namespace = _, boxwhisker) : namespace; }; + hm.namespace = function(_) { return arguments.length ? (namespace = _, hm) : namespace; }; /** * Gets / sets the objectClass - * (see {@link boxwhisker#objectClass}) + * (see {@link heatmap#objectClass}) * @param {string} [_=none] - * @returns {boxwhisker | string} - * @memberof boxwhisker + * @returns {heatmap | string} + * @memberof heatmap * @property * by default objectClass = 'tick-group' */ - boxwhisker.objectClass = function(_) { return arguments.length ? (objectClass = _, boxwhisker) : objectClass; }; + hm.objectClass = function(_) { return arguments.length ? (objectClass = _, hm) : objectClass; }; /** * Gets / sets the transitionDuration - * (see {@link boxwhisker#transitionDuration}) + * (see {@link heatmap#transitionDuration}) * @param {number} [_=none] - * @returns {boxwhisker | number} - * @memberof boxwhisker + * @returns {heatmap | number} + * @memberof heatmap * @property * by default transitionDuration = 1000 */ - boxwhisker.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, boxwhisker) : transitionDuration; }; + hm.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, hm) : transitionDuration; }; /** * Gets / sets the easeFunc - * (see {@link boxwhisker#easeFunc}) + * (see {@link heatmap#easeFunc}) * @param {d3.ease} [_=none] - * @returns {boxwhisker | d3.ease} - * @memberof boxwhisker + * @returns {heatmap | d3.ease} + * @memberof heatmap * @property * by default easeFunc = d3.easeExp */ - boxwhisker.easeFunc = function(_) { return arguments.length ? (easeFunc = _, boxwhisker) : easeFunc; }; + hm.easeFunc = function(_) { return arguments.length ? (easeFunc = _, hm) : easeFunc; }; /** - * Gets / sets the barKeys - * (see {@link boxwhisker#boxKeys}) - * @param {string[]} [_=none] - * @returns {boxwhisker | string[]} - * @memberof boxwhisker + * Gets / sets the tooltip + * (see {@link heatmap#tooltip}) + * @param {tooltip} [_=none] + * @returns {heatmap | tooltip} + * @memberof heatmap * @property - * by default boxKeys = undefined + * by default tooltip = tooltip() */ - boxwhisker.boxKeys = function(_) { return arguments.length ? (boxKeys = _, boxwhisker) : boxKeys; }; + hm.tooltip = function(_) { return arguments.length ? (tooltip$$1 = _, hm) : tooltip$$1; }; + /** - * Gets / sets the boxValues - * (see {@link boxwhisker#boxValues}) - * @param {number[]} [_=none] - * @returns {boxwhisker | number[]} - * @memberof boxwhisker + * Gets / sets the colorFunction + * (see {@link heatmap#colorFunction}) + * @param {colorFunction} [_=none] + * @returns {heatmap | colorFunction} + * @memberof heatmap * @property - * by default boxValues = undefined + * by default colorFunction = colorFunction() */ - boxwhisker.boxValues = function(_) { return arguments.length ? (boxValues = _, boxwhisker) : boxValues; }; + hm.colorFunction = function(_) { return arguments.length ? (colorFunction$$1 = _, hm) : colorFunction$$1; }; + /** - * Gets / sets the objectSize - * (see {@link boxwhisker#objectSize}) + * Gets / sets the xSize + * (see {@link heatmap#xSize}) * @param {number} [_=none] - * @returns {boxwhisker | number} - * @memberof boxwhisker + * @returns {heatmap | number} + * @memberof heatmap * @property - * by default objectSize = undefined + * by default xSize = undefined */ - boxwhisker.objectSize = function(_) { return arguments.length ? (objectSize = _, boxwhisker) : objectSize; }; + hm.xSize = function(_) { return arguments.length ? (xSize = _, hm) : xSize; }; /** - * Gets / sets the spacerSize - * (see {@link boxwhisker#spacerSize}) + * Gets / sets the xSpacerSize + * (see {@link heatmap#xSpacerSize}) * @param {number} [_=none] - * @returns {boxwhisker | number} - * @memberof boxwhisker + * @returns {heatmap | number} + * @memberof heatmap * @property - * by default spacerSize = undefined + * by default xSpacerSize = undefined */ - boxwhisker.spacerSize = function(_) { return arguments.length ? (spacerSize = _, boxwhisker) : spacerSize; }; + hm.xSpacerSize = function(_) { return arguments.length ? (xSpacerSize = _, hm) : xSpacerSize; }; /** - * Gets / sets the tooltip - * (see {@link boxwhisker#tooltip}) - * @param {tooltip} [_=none] - * @returns {boxwhisker | tooltip} - * @memberof boxwhisker + * Gets / sets the ySize + * (see {@link heatmap#ySize}) + * @param {number} [_=none] + * @returns {heatmap | number} + * @memberof heatmap * @property - * by default tooltip = tooltip() + * by default ySize = undefined + */ + hm.ySize = function(_) { return arguments.length ? (ySize = _, hm) : ySize; }; + /** + * Gets / sets the ySpacerSize + * (see {@link heatmap#ySpacerSize}) + * @param {number} [_=none] + * @returns {heatmap | number} + * @memberof heatmap + * @property + * by default ySpacerSize = undefined */ - boxwhisker.tooltip = function(_) { return arguments.length ? (tooltip$$1 = _, boxwhisker) : tooltip$$1; }; + hm.ySpacerSize = function(_) { return arguments.length ? (ySpacerSize = _, hm) : ySpacerSize; }; + // hm.yKeySortingFunction = function(_) { return arguments.length ? (yKeySortingFunction = _, hm) : yKeySortingFunction; } + // hm.xKeySortingFunction = function(_) { return arguments.length ? (xKeySortingFunction = _, hm) : xKeySortingFunction; } - function boxwhisker() { - // for convenience in handling orientation specific values - var horizontalQ = (orient == 'horizontal') ? true : false; - var verticalQ = !horizontalQ; - // background cliping rectangle + function hm() { var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; - var container = setupContainer( selection, namespace, bgcpRect, backgroundFill ); - - // if grouping is undefined sort keys by sorting funct - var ordered = (grouping == undefined) ? d3.keys(data).sort(sortingFunction) : grouping; - // to prevent re-calculation and getters to be passed to axes - boxKeys = flatten(ordered); - boxValues = boxKeys.map(valueExtractor); - - - var numberOfObjects = boxKeys.length; - var extent = [ - Math.min(...boxValues.map(function(d,i){return d[quartilesKeys[0]]})) - domainPadding, - Math.max(...boxValues.map(function(d,i){return d[quartilesKeys[4]]})) + domainPadding - ]; - - // set the scale - scale.domain(extent).range(horizontalQ ? [0,spaceY] : [spaceX, 0]); - var space = horizontalQ ? spaceX : spaceY; - // calculate object size - objectSize = calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ); - // calculate spacer size if needed - spacerSize = calculateWidthOfSpacer(boxKeys, space, objectSize, numberOfObjects, objectSpacer, overflowQ); - // make the nested groups - var spacerFunction = groupingSpacer() - .horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects) - .objectClass(objectClass).objectSize(objectSize).spacerSize(spacerSize) - .transitionDuration(transitionDuration).easeFunc(easeFunc) - .namespace(namespace); - - // move stuff - spacerFunction(container, ordered, 0); - - var parentIndexArray = []; - container.selectAll('g:not(.to-remove).'+objectClass) - .each(function(d, i){if (hasQ(boxKeys, d)){ parentIndexArray.push(Number(d3.select(this).attr('parent-index')));}}); + var container = utils$1.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); + cellKeys = d3.keys(data); + xValues = utils$1.arr.unique(cellKeys.map(xExtractor)); + yValues = utils$1.arr.unique(cellKeys.map(yExtractor)); + vValues = utils$1.arr.unique(cellKeys.map(vExtractor)); - colorFunction$$1 = colorFunction$$1.colorBy() == 'index' - ? colorFunction$$1.dataExtent([0, Math.max(...parentIndexArray)]) - : colorFunction$$1.dataExtent(extent); - - - // set attributes for box and whiskers - container.selectAll('g:not(.to-remove).'+objectClass).each(function(key, i) { - var t = d3.select(this), - currentData = data[key], - - quartiles$$1 = valueExtractor(key, i), - q0 = quartiles$$1[quartilesKeys[0]], - q1 = quartiles$$1[quartilesKeys[1]], - q2 = quartiles$$1[quartilesKeys[2]], - q3 = quartiles$$1[quartilesKeys[3]], - q4 = quartiles$$1[quartilesKeys[4]]; - - var i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'), - fillColor = colorFunction$$1(key, q2, i, 'fill'), // prevent duplicate computation - strokeColor = colorFunction$$1(key, q2, i, 'stroke'); - - - var - whisk = safeSelect(t, 'g', 'whisker'), - uWhisk = safeSelect(whisk, 'path', 'upper'), - lWhisk = safeSelect(whisk, 'path', 'lower'), - quart = safeSelect(t, 'g', 'quartile'), - uQuart = safeSelect(quart, 'rect', 'upper'), - lQuart = safeSelect(quart, 'rect', 'lower'), - mQuart = safeSelect(quart, 'circle', 'median'); - - - // set upper quartile (q3) - uQuart.transition().duration(transitionDuration).ease(easeFunc) - .attr('width', horizontalQ ? objectSize : scale(q3) - scale(q2)) - .attr('height', verticalQ ? objectSize : scale(q3) - scale(q2)) - .attr('fill', fillColor) - .attr('stroke', strokeColor) - .attr('stroke-width', boxStrokeWidth) - .attr('transform', function(d, i){ - var - x = horizontalQ ? 0 : scale(q2), - y = verticalQ ? 0 : scale(extent[1]) - scale(q3), - t = 'translate('+x+','+y+')'; - return t - }); - - // set lower quartile (q1) - lQuart.transition().duration(transitionDuration).ease(easeFunc) - .attr('width', horizontalQ ? objectSize : scale(q2) - scale(q1)) - .attr('height', verticalQ ? objectSize : scale(q2) - scale(q1)) - .attr('fill', fillColor) - .attr('stroke', strokeColor) - .attr('stroke-width', boxStrokeWidth) - .attr('transform', function(d, i){ - var - x = horizontalQ ? 0 : scale(q1), - y = verticalQ ? 0 : scale(extent[1]) - scale(q2), - t = 'translate('+x+','+y+')'; - return t - }); - + cellKeys.sort(function(a, b){ return xKeySortingFunction(a, b) || yKeySortingFunction(a, b) }); + utils$1.con.log('heatmap', 'cells are sorted by', cellKeys); - // set median (q2) - mQuart.transition().duration(transitionDuration).ease(easeFunc) - .attr('r', function(d, i){ - var r = objectSize / 2; - var dif = (scale(q3) - scale(q1)) / 2; - return (r > dif) ? dif : r - }) - .attr('fill', fillColor) - .attr('stroke', strokeColor) - .attr('stroke-width', boxStrokeWidth) - .attr('transform', function(d, i){ - var - x = horizontalQ ? objectSize / 2 : scale(q2), - y = verticalQ ? objectSize / 2 : scale(extent[1]) - scale(q2), - t = 'translate('+x+','+y+')'; - return t - }); - // set lower whisker (min) - lWhisk.transition().duration(transitionDuration).ease(easeFunc) - .attr('d', function(dd, ii){ - var - dir = false, - x = 0, - y = 0, - h = horizontalQ ? scale(q1) - scale(q0) : objectSize, - w = verticalQ ? scale(q1) - scale(q0) : objectSize; - return whiskerPath(dir, x, y, w, h, whiskerWidthPercent, orient) - }) - .attr('transform', function(d, i){ - var - x = horizontalQ ? 0 : scale(q1), - y = verticalQ ? 0 : scale(extent[1]) - scale(q1), - t = 'translate('+x+','+y+')'; - return t - }) - .attr('stroke', 'black').attr('stroke-width', whiskerStrokeWidth) - .attr('fill', 'none'); - // set upper whisker (max) - uWhisk.transition().duration(transitionDuration).ease(easeFunc) - .attr('d', function(dd, ii){ - var - dir = true, - x = 0, - y = 0, - h = horizontalQ ? scale(q4) - scale(q3) : objectSize, - w = verticalQ ? scale(q4) - scale(q3) : objectSize; - return whiskerPath(dir, x, y, w, h, whiskerWidthPercent, orient) - }) - .attr('transform', function(d, i){ - var - x = horizontalQ ? 0 : scale(q3), - y = verticalQ ? 0 : scale(extent[1]) - scale(q4), - t = 'translate('+x+','+y+')'; - return t - }) - .attr('stroke', 'black') - .attr('stroke-width', whiskerStrokeWidth) - .attr('fill', 'none'); + utils$1.con.log('heatmap', 'x and y keys are', {x: xValues, y:yValues}); - }); - tooltip$$1.selection(container.selectAll('g:not(.to-remove).'+objectClass)) - .data(data); - tooltip$$1(); + var xDim = xValues.length, yDim = yValues.length; - } + ySize = utils$1.math.calculateWidthOfObject(spaceY, yDim, yMinObjectSize, yMaxObjectSize, objectSpacer, overflowQ); + xSize = utils$1.math.calculateWidthOfObject(spaceX, xDim, xMinObjectSize, xMaxObjectSize, objectSpacer, overflowQ); + ySpacerSize = utils$1.math.calculateWidthOfSpacer(yValues, spaceY, ySize, yDim, objectSpacer, overflowQ); + xSpacerSize = utils$1.math.calculateWidthOfSpacer(xValues, spaceX, xSize, xDim, objectSpacer, overflowQ); + // console.table({ + // x:{ + // object: xSize, + // spacer: xSpacerSize, + // dim: xDim + // }, + // y:{ + // object: ySize, + // spacer: ySpacerSize, + // dim: yDim + // } + // + // }) + utils$1.con.log('heatmap', 'size of', {x: xSize, y: ySize}); - return boxwhisker -} -function selectFilter(selection) { - var - data, - namespace = 'd3sm-select-filter', - selectionName = 'Select options:', - defaultValue = undefined; + var ySpacer = groupingSpacer() + .horizontalQ(false) + .moveby('category') + .numberOfObjects(yDim) + .objectClass(utils$1.str.hypenate(objectClass, 'row')) + .objectSize(ySize + ySpacerSize) + .spacerSize(0) + .transitionDuration(transitionDuration) + .easeFunc(easeFunc) + .namespace('row'); + var xSpacer = groupingSpacer() + .horizontalQ(true) + .moveby('category') + .numberOfObjects(xDim) + .objectClass(objectClass) + .objectSize(xSize + xSpacerSize) + .spacerSize(0) + .transitionDuration(transitionDuration) + .easeFunc(easeFunc); + ySpacer(container, yValues, 0); + container.selectAll('g.'+utils$1.str.hypenate(objectClass, 'row')) + .each(function(d, i){ xSpacer(d3.select(this), xValues, 0); }); - var lastValue = undefined; + var cells = container.selectAll('g:not(.to-remove).'+objectClass); - selectFilter.data = function(_) { return arguments.length ? (data = _, selectFilter) : data}; - selectFilter.namespace = function(_) { return arguments.length ? (namespace = _, selectFilter) : namespace}; - selectFilter.selectionName = function(_) { return arguments.length ? (selectionName = _, selectFilter) : selectionName}; - selectFilter.defaultValue = function(_) { return arguments.length ? (defaultValue = _, selectFilter) : defaultValue}; - selectFilter.currentOption = currentOption; - function selectFilter() { - var - container = safeSelect(selection, 'div', 'input-group').classed(hypenate(namespace,'container'),true), + if (cellKeys.length != yValues.length * xValues.length) { + var lookup = {}; + cellKeys.map(function(k, i){ + lookup[xExtractor(k)+'::'+yExtractor(k)] = k; + }); - selectPrepend = safeSelect(container, 'div', 'select-prepend').classed('input-group-prepend', true), - selectPrependSpan = safeSelect(selectPrepend, 'span', 'input-group-text').text(selectionName), + var positionedCellKeys = []; + for (var i = 0; i < yValues.length; i++) { + for (var j = 0; j < xValues.length; j++) { + var lookupValue = lookup[xValues[j]+"::"+yValues[i]]; + if (lookupValue == undefined) { + positionedCellKeys.push(undefined); + } else { + positionedCellKeys.push(lookupValue); + } + } + } - select = safeSelect(container, 'select', 'custom-select').classed(hypenate(namespace,'select'),true), + cells.data(positionedCellKeys); - selectAppend = safeSelect(container, 'div', 'select-append').classed('input-group-prepend', true), - selectAppendButton = safeSelect(selectAppend, 'a', 'filter-button').classed('btn btn-outline-secondary', true), - filterButtonIcon = safeSelect(selectAppendButton, 'i', 'fa fa-filter'), + // maybe breaks this + // !!!!!! IMPORTANT NOTE TODO LOOK HERE BUG + cellKeys = positionedCellKeys; + } else { + cells.data(cellKeys); + } - inputGroup = safeSelect(container, 'div', 'filter-input-group').classed('input-group',true).classed('d-none', true), - inputPrepend = safeSelect(inputGroup, 'div', 'input-group-prepend'), - inputPrependSpan = safeSelect(inputPrepend, 'span', 'input-group-text').classed('search-button', true), - inputPrependSpanIcon = safeSelect(inputPrependSpan,'i','fa fa-search'), - input = safeSelect(inputGroup, 'input', 'form-control').attr('placeholder', 'all').attr('type', 'text'), - inputAppend = safeSelect(inputGroup, 'div', 'input-group-append'), - inputAppendButton = safeSelect(inputAppend, 'a', 'close-button').classed('btn btn-outline-secondary', true), - inputAppendButtonIcon = safeSelect(inputAppendButton, 'i', 'fa fa-close'); + var parentIndexArray = []; + cells.each(function(d, i){ parentIndexArray.push(Number(d3.select(this).attr('parent-index'))); }); - var keys = d3.keys(data), - options = select.selectAll('option'); + colorFunction$$1 = colorFunction$$1.colorBy() == 'index' + ? colorFunction$$1.dataExtent([0, Math.max(...parentIndexArray)]) + : colorFunction$$1.dataExtent([0, Math.max(...vValues)]); - options = options.data(d3.keys(data)); - options = options.merge(options.enter().append('option')) - .attr('value', function(d, i){return d}) - .text(function(d, i){return d}); + cells.each(function(key, i) { + utils$1.con.log('heatmap', 'each cell', {key: key, index: i, node: d3.select(this).node()}); - var - filterButton = selectAppendButton, - closeButton = inputAppendButton; + var t = d3.select(this); + if (key == undefined) {return} + var currentData = data[key], + value = vExtractor(key, i), + i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'), + fillColor = colorFunction$$1(key, value, i, 'fill'), // prevent duplicate computation + strokeColor = colorFunction$$1(key, value, i, 'stroke'); - filterButton.on('click', function(d, i){ - var currentStyle = inputGroup.classed('d-none'); - inputGroup.classed('d-none', !currentStyle); - }); + var c = utils$1.sel.safeSelect(t, 'rect', utils$1.str.hypenate(objectClass,'rect')); + c.attr('width', xSize + xSpacerSize - objectStrokeWidth) + .attr('height', ySize + ySpacerSize - objectStrokeWidth) + .attr('fill', fillColor) + .attr('x', objectStrokeWidth/2) + .attr('y', objectStrokeWidth/2) + .attr('stroke', "#000") + .attr('stroke-width', objectStrokeWidth); - closeButton.on('click', function(d, i){ - input.property('value', '').dispatch('input'); }); - input.on('input', function(d, i){ - var - val = input.property('value'), - reg = new RegExp(val, 'gi'), - use; - - if (val == '') {use = keys;} - else { - use = []; - d3.keys(data).map(function(option, j){ - var match = option.match(reg); - if (match == null || match.join('') == '') ; - else { use.push(option); } - }); - } - - options = select.selectAll('option'); - options = options.data(use); - options.exit().remove(); - options = options.merge(options.enter().append('option')) - .attr('value', function(d, i){return d}) - .text(function(d, i){return d}); - - var current = currentOption(); - if (lastValue != current) { - lastValue = current; - select.dispatch('change'); - } - }); + tooltip$$1.selection(cells.selectAll('rect.'+utils$1.str.hypenate(objectClass, 'rect'))) + .data(data); + // .keys(['r', 'v']) + // .header(function(d, i){return utils.str.hypenate(data[d][xKey], data[d][yKey]) }) + tooltip$$1(); - } - function currentOption() { - var val = selection.select("select").property('value'); - return val == undefined || val == '' - ? defaultValue == undefined - ? d3.keys(data)[0] - : defaultValue - : val } - return selectFilter + return hm; } /******************************************************************************* ** ** ** ** -** DATATOGGLE ** +** VIOLIN ** ** ** ** ** *******************************************************************************/ + /** - * Creates a datatoggle - * @constructor datatoggle + * Creates a violin + * + * {@link https://sumneuron.gitlab.io/d3sm/demos/basic-violins/index.html Demo} + * @constructor violin * @param {d3.selection} selection - * @namespace datatoggle - * @returns {function} datatoggle + * @namespace violin + * @returns {function} violin */ -function datatoggle( selection ) { - var +function violin( selection ) { + var /** - * Keys to make toggle-able options - * (see {@link datatoggle#keys}) - * @param {string[]} [keys=undefined] - * @memberof datatoggle# + * Data to plot. Assumed to be a object, where each key corresponds to a violin + * (see {@link violin#data}) + * @param {Object} [data=undefined] + * @memberof violin# * @property */ - keys, - + data, /** - * What to do when a different key is clicked - * (see {@link datatoggle#updateFunction}) - * @param {function} [updateFunction=function(){}] - * @memberof datatoggle# + * Which direction to render the bars in + * (see {@link violin#orient}) + * @param {number} [orient='horizontal'] + * @memberof violin# * @property */ - updateFunction = function(){}, + orient='horizontal', /** - * Namespace for all items made by this instance of datatoggle - * @param {string} [namespace="d3sm-databar"] - * @memberof datatoggle# + * Amount of horizontal space (in pixels) avaible to render the violin in + * (see {@link violin#spaceX}) + * @param {number} [spaceX=undefined] + * @memberof violin# * @property */ - namespace='d3sm-databar', + spaceX, /** - * Currently toggled key - * @param {string} [currentKey=undefined] - * @memberof datatoggle# + * Amount of vertical space (in pixels) avaible to render the violin in + * (see {@link violin.spaceY}) + * @param {number} [spaceY=undefined] + * @memberof violin# * @property */ - currentKey, - - - - xAxisSelectQ = false, - xAxisOptions, - yAxisSelectQ = false, - yAxisOptions, - data; - toggle.xAxisSelectQ = function(_){return arguments.length ? (xAxisSelectQ = _, toggle) : xAxisSelectQ}; - toggle.yAxisSelectQ = function(_){return arguments.length ? (yAxisSelectQ = _, toggle) : yAxisSelectQ}; - toggle.xAxisOptions = function(_){return arguments.length ? (xAxisOptions = _, toggle) : xAxisOptions}; - toggle.yAxisOptions = function(_){return arguments.length ? (yAxisOptions = _, toggle) : yAxisOptions}; - toggle.data = function(_){return arguments.length ? (data = _, toggle) : data}; - toggle.keys = function(_){return arguments.length ? (keys = _, toggle) : keys}; - toggle.currentKey = function(_){return arguments.length ? (currentKey = _, toggle) : currentKey}; - - /** - * Gets / sets the updateFunction - * (see {@link datatoggle#updateFunction}) - * @function datatoggle.updateFunction - * @param {function} [_=none] - * @returns {datatoggle | function} - * @memberof datatoggle - * @property - * by default updateFunction = function(){} - */ - toggle.updateFunction = function(_){return arguments.length ? (updateFunction = _, toggle) : updateFunction}; + spaceY, /** - * Gets / sets the namespace - * (see {@link datatoggle#namespace}) - * @function datatoggle.namespace - * @param {string} [_=none] - * @returns {datatoggle | string} - * @memberof datatoggle - * @property - * by default namespace = 'd3sm-databar' - */ - toggle.namespace = function(_){return arguments.length ? (namespace = _, toggle) : namespace}; + * Whether or not to allow violin to render elements pass the main spatial dimension + * given the orientation (see {@link violin#orient}), where {@link violin#orient}="horizontal" + * the main dimension is {@link violin#spaceX} and where {@link violin#orient}="vertical" + * the main dimension is {@link violin#spaceY} + * @param {boolean} [overflowQ=false] + * @memberof violin# + * @property + */ + overflowQ = true, /** - * Gets / sets the currentKey - * (see {@link datatoggle#currentKey}) - * @function datatoggle.currentKey - * @param {string} [_=none] - * @returns {datatoggle | string} - * @memberof datatoggle - * @property - * by default currentKey = undefined - */ - toggle.currentKeys = - function () { - var vals = {}; - d3.keys(filterSelects).map(function(k, i){ - vals[k]= filterSelects[k].currentOption(); - }); - return vals - }; - - var filterSelects = {}; - function toggle() { - // selection options - - // selection.classed('d-flex flex-row', true) - // var filterButton = safeSelect(selection, 'a', 'slider-buttons') - // var filterI = safeSelect(filterButton, 'i', 'fa fa-sliders') - - /*BUG: unexpected behavior. - - when using bootstrap-eque way for collapse, clicking button submits to - the same page in applications (but not in demo), so using anchor () - - when using anchor, cause page to jump to that location - - when using show, the first open works, but the close does not. - */ - // filterButton.attr('class', 'btn btn-secondary') - // .attr('data-toggle', 'collapse') - // .attr('href', '#'+hypenate(namespace, 'data-toggle')) - // .attr('target', '_blank') - // .html(filterButton.html()+' Filters') - // .on('click', function(d, i){ - // d3.event.preventDefault() - // d3.event.stopPropagation() - // var dt = d3.select("#"+hypenate(namespace, 'data-toggle')) - // dt.classed('show', !dt.classed('show')) - // dt.classed('d-inline-flex', dt.classed('show')) - // - // filterButton.classed('btn-primary', dt.classed('show')) - // filterButton.classed('btn-secondary', !dt.classed('show')) - // }) - // .classed('d-inline-flex', true) - // .style("margin-right", '10px') - - - - // var datatoggleCollapse = safeSelect(selection, 'div', hypenate(namespace,'collapse')) - // .attr('id', hypenate(namespace, 'data-toggle')) - // .classed('collapse', true) - - // var flexRow = safeSelect(datatoggleCollapse, 'div', 'd-inline-flex flex-row flex-wrap') - var flexRow = safeSelect(selection, 'div', 'd-inline-flex flex-row flex-wrap'); - - var dataopts = flexRow.selectAll('div.'+hypenate(namespace,'select-filter')); - // remove excess - dataopts.exit().remove(); - // bind data - dataopts = dataopts.data(d3.keys(data)); - //enter - var doEnter = dataopts.enter().append('div') - .attr('class', 'select-filter'); - - dataopts = dataopts.merge(doEnter).style('margin-right', "10px"); - - dataopts.each(function(d, i){ - var t = d3.select(this); - var sf = selectFilter(t) - .data(data[d]) - .namespace(hypenate(namespace, d)) - .selectionName(d); - sf(); - filterSelects[d] = sf; - }); - - - selection.selectAll('select') - .on('change', function(){updateFunction();}); - // bind update function - // d3.selectAll - return toggle - } - - return toggle -} - -/******************************************************************************* -** ** -** ** -** SCATTER ** -** ** -** ** -*******************************************************************************/ -/** - * Creates a scatter - * - * {@link https://sumneuron.gitlab.io/d3sm/demos/scatter/index.html Demo} - * @constructor scatter - * @param {d3.selection} selection - * @namespace scatter - * @returns {function} scatter - */ -function scatter ( selection ) { - - var - /** - * Data to plot. Assumed to be a object, where each key corresponds to a point - * (see {@link scatter#data}) - * @param {Object} [data=undefined] - * @memberof scatter# + * Whether or not to display points inside the points + * @param {boolean} [pointsQ=false] + * @memberof violin# * @property */ - data, + pointsQ = true, /** - * Amount of horizontal space (in pixels) avaible to render the scatter in - * (see {@link scatter#spaceX}) - * @param {number} [spaceX=undefined] - * @memberof scatter# + * An array - putatively of other arrays - depicting how bars should be arranged + * @param {Array[]} [grouping=undefined] + * @memberof violin# * @property */ - spaceX, + grouping, /** - * Amount of vertical space (in pixels) avaible to render the scatter in - * (see {@link scatter.spaceY}) - * @param {number} [spaceY=undefined] - * @memberof scatter# + * How to get the value of the violin + * @param {function} [valueExtractor=function(key, index) { return data[key] }] + * @memberof violin# * @property */ - spaceY, - + valueExtractor = function(key, index) {return data[key] }, /** - * The scale for which scatter x values should be transformed by - * @param {d3.scale} [scaleX=d3.scaleLinear] - * @memberof scatter# + * How to sort the bars - if {@link violin#grouping} is not provided. + * @param {function} [sortingFunction=function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}] + * @memberof violin# * @property */ - scaleX = d3.scaleLinear(), + sortingFunction = function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}, + /** - * The padding for the domain of the scaleX (see {@link scatter#scaleX}) - * @param {number} [domainPaddingX=0.5] - * @memberof scatter# + * The scale for which violin values should be transformed by + * @param {d3.scale} [scale=d3.scaleLinear] + * @memberof violin# * @property */ - domainPaddingX = 0.5, + scale = d3.scaleLinear(), /** - * The function for getting the x value of the current point - * @param {function} [valueExtractorX=function(d, i){return data[d]['x']}] - * @memberof scatter# + * The padding for the domain of the scale (see {@link violin#scale}) + * @param {number} [domainPadding=0.5] + * @memberof violin# * @property */ - valueExtractorX = function(d, i) {return data[d]['x']}, - + domainPadding = 0.5, /** - * The scale for which scatter y values should be transformed by - * @param {d3.scale} [scaleY=d3.scaleLinear] - * @memberof scatter# + * Default space for the spacer (percentage) of main dimension given the orientation + * (see {@link violin#orient}), where {@link violin#orient}="horizontal" + * the main dimension is {@link violin#spaceX} and where {@link violin#orient}="vertical" + * the main dimension is {@link violin#spaceY} between bars + * @param {number} [objectSpacer=0.05] + * @memberof violin# * @property */ - scaleY = d3.scaleLinear(), + objectSpacer = 0.05, /** - * The padding for the domain of the scaleY (see {@link scatter#scaleY}) - * @param {number} [domainPaddingY=0.5] - * @memberof scatter# + * The minimum size that an object can be + * @param {number} [minObjectSize=50] + * @memberof violin# * @property */ - domainPaddingY = 0.5, + minObjectSize = 50, /** - * The function for getting the y value of the current point - * @param {function} [valueExtractorY=function(d, i){return data[d]['y']}] - * @memberof scatter# + * The maximum size that an object can be + * @param {number} [maxObjectSize=100] + * @memberof violin# * @property */ - valueExtractorY = function(d, i) {return data[d]['y']}, - + maxObjectSize = 100, /** - * The scale for which scatter r values should be transformed by - * @param {d3.scale} [scaleR=d3.scaleLinear] - * @memberof scatter# - * @property - */ - scaleR = d3.scaleLinear(), - /** - * The padding for the domain of the scaleR (see {@link scatter#scaleR}) - * @param {number} [domainPaddingR=0.5] - * @memberof scatter# + * The stroke width of the bars + * @param {number} [barStrokeWidth=2] + * @memberof violin# * @property */ - domainPaddingR = 0.5, + objectStrokeWidth = 2, /** - * The function for getting the r value of the current point - * @param {function} [valueExtractorR=function(d, i){return 2}] - * @memberof scatter# + * Instance of ColorFunction + * @param {function} [colorFunction = colorFunction()] + * @memberof violin# * @property */ - valueExtractorR = function(d, i) {return 2}, + colorFunction$$1 = colorFunction(), /** - * The min radius a point can have - * @param {function} [minRadius=2] - * @memberof scatter# + * Instance of ColorFunction modified by a scale for the points + * @param {function} [pointColorFunc = colorFunction()] + * @memberof violin# * @property */ - minRadius = 2, + pointColorFunc = function (d, type, base, min, max) { + var minMaxHexScale = d3.scaleLinear().domain([min, max]).range([-0.25, 0.05]); + var scaledColor = utils.color.modifyHexidecimalColorLuminance(base.replace('#', ''), minMaxHexScale(d)); + var mod = type == "stroke" ? 0 : 0.25; + return utils.color.modifyHexidecimalColorLuminance(scaledColor.replace('#', ''), mod) + }, + /** - * The min radius a point can have - * @param {function} [maxRadius=10] - * @memberof scatter# + * The radius of a point + * @param {number} [pointRadius=3] + * @memberof violin# * @property */ - maxRadius = 10, - + pointRadius = 3, /** - * The stroke width of the points + * The stroke width of the oints * @param {number} [pointStrokeWidth=2] - * @memberof scatter# + * @memberof violin# * @property */ pointStrokeWidth = 2, - /** - * Instance of ColorFunction - * @param {function} [colorFunction = colorFunction()] - * @memberof scatter# - * @property - */ - colorFunction$$1 = colorFunction(), + /** * Color of the background * @param {string} [backgroundFill="transparent"] - * @memberof scatter# + * @memberof violin# * @property */ backgroundFill = 'transparent', /** - * Namespace for all items made by this instance of scatter - * @param {string} [namespace="d3sm-scatter"] - * @memberof scatter# + * Namespace for all items made by this instance of violin + * @param {string} [namespace="d3sm-violin"] + * @memberof violin# * @property */ - namespace = 'd3sm-scatter', + namespace = 'd3sm-violin', /** - * Class name for scatter container ( element) - * @param {string} [objectClass="scatter-point"] - * @memberof scatter# + * Class name for violin container ( element) + * @param {string} [objectClass="violin"] + * @memberof violin# * @property */ - objectClass = 'scatter-point', + objectClass = 'violin', /** * Duration of all transitions of this element * @param {number} [transitionDuration=1000] - * @memberof scatter# + * @memberof violin# * @property */ transitionDuration = 1000, /** * Easing function for transitions * @param {d3.ease} [easeFunc=d3.easeExp] - * @memberof scatter# + * @memberof violin# * @property */ easeFunc = d3.easeExp, - // useful values to extract to prevent re-calculation /** - * The keys of the points - * @param {string[]} [pointKeys=undefined] - * @memberof scatter# + * The keys corresponding to each quartile + * @param {string[]} [quartileKeys=["Q0", "Q1", "Q2", "Q3", "Q4"]] + * @memberof violin# * @property */ - pointKeys, + quartileKeys = ["Q0", "Q1", "Q2", "Q3", "Q4"], + /** - * The x values of the points - * @param {number[]} [valuesX=undefined] - * @memberof scatter# + * The keys of the bars + * @param {string[]} [violinKeys=undefined] + * @memberof violin# * @property */ - valuesX, + violinKeys, /** - * The y values of the points - * @param {number[]} [valuesY=undefined] - * @memberof scatter# + * The values of the bars + * @param {number[]} [violinValues=undefined] + * @memberof violin# * @property */ - valuesY, + violinValues, /** - * The r values of the points - * @param {number[]} [valuesR=undefined] - * @memberof scatter# + * The objectSize (actual width) used by the bars + * @param {number} [objectSize=undefined] + * @memberof violin# * @property */ - valuesR, + objectSize, + /** + * The spacerSize (actual width) used by the spacers between the bars + * @param {number} [spacerSize=undefined] + * @memberof violin# + * @property + */ + spacerSize, /** * Instance of Tooltip * @param {function} [tooltip=tooltip()] - * @memberof scatter# + * @memberof violin# * @property */ - tooltip$$1 = tooltip(); + tooltip$$1 = tooltip().keys([quartileKeys[4], quartileKeys[3], quartileKeys[2], quartileKeys[1], quartileKeys[0]]), + pointsTooltip = tooltip(), + // pointKeyExtractor = function(violinKey, violinData, violinValues) {return d3.keys(violinValues[violinKey].values)}, + // pointValueExtractor = function(pointKey, violinKey, violinData, violinValues) {return violinValues[violinKey].values[pointKey]}, + /** - * Gets or sets the selection in which items are manipulated - * @param {d3.selection} [_=none] - * @returns {scatter | d3.selection} - * @memberof scatter + * Function which given the key of the violin and that key's associated value + * returns the object consiting of the points of the violin + * (see {@link violin#violinPointsExtractor}) + * @param {Object} [violinPointsExtractor=function(violinKey, violinData) { return violinData.points }] + * @memberof violin# + * @property + */ + violinPointsExtractor = function (violinKey, violinData) {return violinData.points }, + /** + * Function which given the key of the current point and the object of points for the + * violin, returns the numerical value of the point + * (see {@link violin#violinPointValueExtractor}) + * @param {Object} [violinPointValueExtractor=function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value }] + * @memberof violin# + * @property + */ + violinPointValueExtractor = function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value }; + + + /** + * Gets or sets the violinPointsExtractor + * @param {function} [_=none] + * @returns {violin | function} + * @memberof violin + * @property + * by default violinPointsExtractor = function(violinKey, violinData) { return violinData.points } + */ + violin.violinPointsExtractor = function(_) { return arguments.length ? (violinPointsExtractor = _, violin) : violinPointsExtractor; }; + /** + * Gets or sets the violinPointValueExtractor + * @param {function} [_=none] + * @returns {violin | function} + * @memberof violin + * @property + * by default violinPointsExtractor = function(pointKey, violinKey, violinData, violinValues) {return violinValues[violinKey].values[pointKey]} + */ + violin.violinPointValueExtractor = function(_) { return arguments.length ? (violinPointValueExtractor = _, violin) : violinPointValueExtractor; }; + + + /** + * Gets or sets the selection in which items are manipulated + * @param {d3.selection} [_=none] + * @returns {violin | d3.selection} + * @memberof violin * @property * by default selection = selection */ - scatter.selection = function(_) { return arguments.length ? (selection =_, scatter) : selection}; + violin.selection = function(_) { return arguments.length ? (selection = _, violin) : selection; }; /** * Gets or sets the data - * (see {@link scatter#data}) + * (see {@link violin#data}) * @param {number} [_=none] - * @returns {scatter | object} - * @memberof scatter + * @returns {violin | object} + * @memberof violin * @property */ - scatter.data = function(_) { return arguments.length ? (data =_, scatter) : data}; + violin.data = function(_) { return arguments.length ? (data = _, violin) : data; }; + /** + * Gets or sets the orient of the boxes + * (see {@link violin#orient}) + * @param {number} [_=none] + * @returns {violin | object} + * @memberof violin + * @property + */ + violin.orient = function(_) { return arguments.length ? (orient = _, violin) : orient; }; /** * Gets or sets the amount of horizontal space in which items are manipulated - * (see {@link scatter#spaceX}) + * (see {@link violin#spaceX}) * @param {number} [_=none] should be a number > 0 - * @returns {scatter | number} - * @memberof scatter + * @returns {violin | number} + * @memberof violin * @property * by default spaceX = undefined */ - scatter.spaceX = function(_) { return arguments.length ? (spaceX =_, scatter) : spaceX}; + violin.spaceX = function(_) { return arguments.length ? (spaceX = _, violin) : spaceX; }; /** * Gets or sets the amount of vertical space in which items are manipulated - * (see {@link scatter#spaceY}) + * (see {@link violin#spaceY}) * @param {number} [_=none] should be a number > 0 - * @returns {scatter | number} - * @memberof scatter + * @returns {violin | number} + * @memberof violin * @property * by default spaceY = undefined */ - scatter.spaceY = function(_) { return arguments.length ? (spaceY =_, scatter) : spaceY}; - + violin.spaceY = function(_) { return arguments.length ? (spaceY = _, violin) : spaceY; }; /** - * Gets / sets the x scale for which the scatter x values should be transformed by - * (see {@link scatter#scaleX}) - * @param {d3.scale} [_=none] - * @returns {scatter | d3.scale} - * @memberof scatter - * @property - * by default scaleX = d3.scaleLinear() - */ - scatter.scaleX = function(_) { return arguments.length ? (scaleX =_, scatter) : scaleX}; - /** - * Gets / sets the padding for the domain of the x scale - * (see {@link scatter#domainPaddingX}) - * @param {number} [_=none] - * @returns {scatter | number} - * @memberof scatter + * Gets / sets whether or not violin is allowed to go beyond specified dimensions + * (see {@link violin#overflowQ}) + * @param {boolean} [_=none] + * @returns {violin | boolean} + * @memberof violin * @property - * by default domainPaddingX = 0.5 + * by default overflowQ = false */ - scatter.domainPaddingX = function(_) { return arguments.length ? (domainPaddingX =_, scatter) : domainPaddingX}; + violin.overflowQ = function(_) { return arguments.length ? (overflowQ = _, violin) : overflowQ; }; /** - * Gets / sets the valueExtractorX - * (see {@link scatter#valueExtractorX}) - * @param {function} [_=none] - * @returns {scatter | function} - * @memberof scatter + * Gets / sets whether or not to plot points with the violins + * (see {@link violin#pointsQ}) + * @param {boolean} [_=none] + * @returns {violin | boolean} + * @memberof violin * @property - * by default valueExtractorX = function(key, index) { return data[key]['x'] } + * by default pointsQ = false */ - scatter.valueExtractorX = function(_) { return arguments.length ? (valueExtractorX =_, scatter) : valueExtractorX}; + violin.pointsQ = function(_) { return arguments.length ? (pointsQ = _, violin) : pointsQ; }; /** - * Gets / sets the y scale for which the scatter y values should be transformed by - * (see {@link scatter#scaleY}) - * @param {d3.scale} [_=none] - * @returns {scatter | d3.scale} - * @memberof scatter + * Gets / sets the grouping of the boxes + * (see {@link violin#grouping}) + * @param {Array[]} [_=none] + * @returns {violin | Array[]} + * @memberof violin * @property - * by default scaleY = d3.scaleLinear() + * by default grouping = undefined */ - scatter.scaleY = function(_) { return arguments.length ? (scaleY =_, scatter) : scaleY}; + violin.grouping = function(_) { return arguments.length ? (grouping = _, violin) : grouping; }; /** - * Gets / sets the padding for the domain of the y scale - * (see {@link scatter#domainPaddingY}) - * @param {number} [_=none] - * @returns {scatter | number} - * @memberof scatter + * Gets / sets the valueExtractor + * (see {@link violin#valueExtractor}) + * @param {function} [_=none] + * @returns {violin | function} + * @memberof violin * @property - * by default domainPaddingY = 0.5 */ - scatter.domainPaddingY = function(_) { return arguments.length ? (domainPaddingY =_, scatter) : domainPaddingY}; + violin.valueExtractor = function(_) { return arguments.length ? (valueExtractor = _, violin) : valueExtractor; }; /** - * Gets / sets the valueExtractorY - * (see {@link scatter#valueExtractorY}) + * Gets / sets the sortingFunction + * (see {@link violin#sortingFunction}) * @param {function} [_=none] - * @returns {scatter | function} - * @memberof scatter + * @returns {violin | function} + * @memberof violin * @property - * by default valueExtractorY = function(key, index) { return data[key]['y'] } */ - scatter.valueExtractorY = function(_) { return arguments.length ? (valueExtractorY =_, scatter) : valueExtractorY}; - + violin.sortingFunction = function(_) { return arguments.length ? (sortingFunction = _, violin) : sortingFunction; }; /** - * Gets / sets the r scale for which the scatter r values should be transformed by - * (see {@link scatter#scaleR}) + * Gets / sets the scale for which the violin values should be transformed by + * (see {@link violin#scale}) * @param {d3.scale} [_=none] - * @returns {scatter | d3.scale} - * @memberof scatter + * @returns {violin | d3.scale} + * @memberof violin * @property - * by default scaleR = d3.scaleLinear() + * by default scale = d3.scaleLinear() */ - scatter.scaleR = function(_) { return arguments.length ? (scaleR =_, scatter) : scaleR}; + violin.scale = function(_) { return arguments.length ? (scale = _, violin) : scale; }; /** - * Gets / sets the padding for the domain of the r scale - * (see {@link scatter#domainPaddingR}) + * Gets / sets the padding for the domain of the scale + * (see {@link violin#domainPadding}) * @param {number} [_=none] - * @returns {scatter | number} - * @memberof scatter + * @returns {violin | number} + * @memberof violin * @property - * by default domainPaddingR = 0.5 + * by default domainPadding = 0.5 */ - scatter.domainPaddingR = function(_) { return arguments.length ? (domainPaddingR =_, scatter) : domainPaddingR}; + violin.domainPadding = function(_) { return arguments.length ? (domainPadding = _, violin) : domainPadding; }; + + /** - * Gets / sets the valueExtractorR - * (see {@link scatter#valueExtractorR}) - * @param {function} [_=none] - * @returns {scatter | function} - * @memberof scatter + * Gets / sets objectSpacer + * (see {@link violin#objectSpacer}) + * @param {number} [_=none] + * @returns {violin | number} + * @memberof violin * @property - * by default valueExtractorR = function(key, index) { return data[key]['r'] } + * by default objectSpacer = 0.05 */ - scatter.valueExtractorR = function(_) { return arguments.length ? (valueExtractorR =_, scatter) : valueExtractorR}; + violin.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, violin) : objectSpacer; }; /** - * Gets / sets the minRadius - * (see {@link scatter#minRadius}) + * Gets / sets the minObjectSize + * (see {@link violin#minObjectSize}) * @param {number} [_=none] - * @returns {scatter | number} - * @memberof scatter + * @returns {violin | number} + * @memberof violin * @property - * by default minRadius = 2 + * by default minObjectSize = 15 */ - scatter.minRadius = function(_) { return arguments.length ? (minRadius =_, scatter) : minRadius}; + violin.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, violin) : minObjectSize; }; /** - * Gets / sets the maxRadius - * (see {@link scatter#maxRadius}) + * Gets / sets the maxObjectSize + * (see {@link violin#maxObjectSize}) * @param {number} [_=none] - * @returns {scatter | number} - * @memberof scatter + * @returns {violin | number} + * @memberof violin * @property - * by default maxRadius = 10 + * by default maxObjectSize = 50 */ - scatter.maxRadius = function(_) { return arguments.length ? (maxRadius =_, scatter) : maxRadius}; + violin.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, violin) : maxObjectSize; }; /** - * Gets / sets the pointStrokeWidth - * (see {@link scatter#pointStrokeWidth}) + * Gets / sets the objectStrokeWidth + * (see {@link violin#objectStrokeWidth}) * @param {number} [_=none] - * @returns {scatter | number} - * @memberof scatter + * @returns {violin | number} + * @memberof violin * @property - * by default pointStrokeWidth = 2 + * by default objectStrokeWidth = 2 */ - scatter.pointStrokeWidth = function(_) { return arguments.length ? (pointStrokeWidth =_, scatter) : pointStrokeWidth}; + violin.objectStrokeWidth = function(_) { return arguments.length ? (objectStrokeWidth = _, violin) : objectStrokeWidth; }; + + /** * Gets / sets the colorFunction - * (see {@link scatter#colorFunction}) - * @param {number} [_=none] - * @returns {scatter | number} - * @memberof scatter + * (see {@link violin#colorFunction}) + * @param {colorFunction} [_=none] + * @returns {violin | colorFunction} + * @memberof violin * @property * by default colorFunction = colorFunction() */ - scatter.colorFunction = function(_) { return arguments.length ? (colorFunction$$1 =_, scatter) : colorFunction$$1}; + violin.colorFunction = function(_) { return arguments.length ? (colorFunction$$1 = _, violin) : colorFunction$$1; }; /** - * Gets / sets the backgroundFill - * (see {@link scatter#backgroundFill}) - * @param {string} [_=none] - * @returns {scatter | string} - * @memberof scatter - * @property - * by default backgroundFill = 'transparent' - */ - scatter.backgroundFill = function(_) { return arguments.length ? (backgroundFill =_, scatter) : backgroundFill}; + * Gets / sets the colorFunction + * (see {@link violin#colorFunction}) + * @param {colorFunction} [_=none] + * @returns {violin | colorFunction} + * @memberof violin + * @property + * by default colorFunction = colorFunction() + */ + violin.pointColorFunc = function(_) { return arguments.length ? (pointColorFunc = _, violin) : pointColorFunc; }; + + + /** + * Gets / sets the pointRadius + * (see {@link violin#pointRadius}) + * @param {number} [_=none] + * @returns {violin | number} + * @memberof violin + * @property + * by default pointRadius = 2 + */ + violin.pointRadius = function(_) { return arguments.length ? (pointRadius = _, violin) : pointRadius; }; + /** + * Gets / sets the pointStrokeWidth + * (see {@link violin#pointStrokeWidth}) + * @param {number} [_=none] + * @returns {violin | number} + * @memberof violin + * @property + * by default pointStrokeWidth = 2 + */ + violin.pointStrokeWidth = function(_) { return arguments.length ? (pointStrokeWidth = _, violin) : pointStrokeWidth; }; + + + /** + * Gets / sets the backgroundFill + * (see {@link violin#backgroundFill}) + * @param {string} [_=none] + * @returns {violin | string} + * @memberof violin + * @property + * by default backgroundFill = 'transparent' + */ + violin.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, violin) : backgroundFill; }; /** * Gets / sets the namespace - * (see {@link scatter#namespace}) + * (see {@link violin#namespace}) * @param {string} [_=none] - * @returns {scatter | string} - * @memberof scatter + * @returns {violin | string} + * @memberof violin * @property - * by default namespace = 'd3sm-scatter' + * by default namespace = 'd3sm-violin' */ - scatter.namespace = function(_) { return arguments.length ? (namespace =_, scatter) : namespace}; + violin.namespace = function(_) { return arguments.length ? (namespace = _, violin) : namespace; }; /** * Gets / sets the objectClass - * (see {@link scatter#objectClass}) + * (see {@link violin#objectClass}) * @param {string} [_=none] - * @returns {scatter | string} - * @memberof scatter + * @returns {violin | string} + * @memberof violin * @property * by default objectClass = 'tick-group' */ - scatter.objectClass = function(_) { return arguments.length ? (objectClass =_, scatter) : objectClass}; + violin.objectClass = function(_) { return arguments.length ? (objectClass = _, violin) : objectClass; }; + + /** * Gets / sets the transitionDuration - * (see {@link scatter#transitionDuration}) + * (see {@link violin#transitionDuration}) * @param {number} [_=none] - * @returns {scatter | number} - * @memberof scatter + * @returns {violin | number} + * @memberof violin * @property * by default transitionDuration = 1000 */ - scatter.transitionDuration = function(_) { return arguments.length ? (transitionDuration =_, scatter) : transitionDuration}; + violin.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, violin) : transitionDuration; }; /** * Gets / sets the easeFunc - * (see {@link scatter#easeFunc}) + * (see {@link violin#easeFunc}) * @param {d3.ease} [_=none] - * @returns {scatter | d3.ease} - * @memberof scatter + * @returns {violin | d3.ease} + * @memberof violin * @property * by default easeFunc = d3.easeExp */ - scatter.easeFunc = function(_) { return arguments.length ? (easeFunc =_, scatter) : easeFunc}; + violin.easeFunc = function(_) { return arguments.length ? (easeFunc = _, violin) : easeFunc; }; + /** - * Gets / sets the pointKeys - * (see {@link scatter#pointKeys}) + * Gets / sets the quartileKey + * (see {@link violin#quartileKey}) + * @param {string} [_=none] + * @returns {violin | string} + * @memberof violin + * @property + * by default quartileKey = "utils.math.quartiles" + */ + violin.quartileKey = function(_) { return arguments.length ? (quartileKey = _, violin) : quartileKey; }; + /** + * Gets / sets the quartileKeys + * (see {@link violin#quartileKeys}) * @param {string[]} [_=none] - * @returns {scatter | string[]} - * @memberof scatter + * @returns {violin | string[]} + * @memberof violin * @property - * by default pointKeys = undefined + * by default quartileKeys = ["Q0","Q1","Q2","Q3","Q4"] */ - scatter.pointKeys = function(_) { return arguments.length ? (pointKeys =_, scatter) : pointKeys}; + violin.quartileKeys = function(_) { return arguments.length ? (quartileKeys = _, violin) : quartileKeys; }; + + /** - * Gets / sets the valuesX - * (see {@link scatter#valuesX}) - * @param {number[]} [_=none] - * @returns {scatter | number[]} - * @memberof scatter + * Gets / sets the violinKeys + * (see {@link violin#violinKeys}) + * @param {string[]} [_=none] + * @returns {violin | string[]} + * @memberof violin * @property - * by default valuesX = undefined + * by default violinKeys = undefined */ - scatter.valuesX = function(_) { return arguments.length ? (valuesX =_, scatter) : valuesX}; + violin.violinKeys = function(_) { return arguments.length ? (violinKeys = _, violin) : violinKeys; }; /** - * Gets / sets the valuesY - * (see {@link scatter#valuesY}) - * @param {number[]} [_=none] - * @returns {scatter | number[]} - * @memberof scatter + * Gets / sets the violinValues + * (see {@link violin#violinValues}) + * @param {Object[]} [_=none] + * @returns {violin | Object[]} + * @memberof violin * @property - * by default valuesY = undefined + * by default violinValues = undefined */ - scatter.valuesY = function(_) { return arguments.length ? (valuesY =_, scatter) : valuesY}; + violin.violinValues = function(_) { return arguments.length ? (violinValues = _, violin) : violinValues; }; + /** - * Gets / sets the valuesR - * (see {@link scatter#valuesR}) - * @param {number[]} [_=none] - * @returns {scatter | number[]} - * @memberof scatter + * Gets / sets the objectSize + * (see {@link violin#objectSize}) + * @param {number} [_=none] + * @returns {violin | number} + * @memberof violin * @property - * by default valuesR = undefined + * by default objectSize = undefined */ - scatter.valuesR = function(_) { return arguments.length ? (valuesR =_, scatter) : valuesR}; + violin.objectSize = function(_) { return arguments.length ? (objectSize = _, violin) : objectSize; }; + /** + * Gets / sets the spacerSize + * (see {@link violin#spacerSize}) + * @param {number} [_=none] + * @returns {violin | number} + * @memberof violin + * @property + * by default spacerSize = undefined + */ + violin.spacerSize = function(_) { return arguments.length ? (spacerSize = _, violin) : spacerSize; }; /** * Gets / sets the tooltip - * (see {@link scatter#tooltip}) + * (see {@link violin#tooltip}) * @param {tooltip} [_=none] - * @returns {scatter | tooltip} - * @memberof scatter + * @returns {violin | tooltip} + * @memberof violin * @property * by default tooltip = tooltip() */ + violin.tooltip = function(_) { return arguments.length ? (tooltip$$1 = _, violin) : tooltip$$1; }; + /** + * Gets / sets the pointsTooltip + * (see {@link violin#pointsTooltip}) + * @param {tooltip} [_=none] + * @returns {violin | tooltip} + * @memberof violin + * @property + * by default pointsTooltip = tooltip() + */ + violin.pointsTooltip = function(_) { return arguments.length ? (pointsTooltip = _, violin) : pointsTooltip; }; - scatter.tooltip = function(_) { return arguments.length ? (tooltip$$1 =_, scatter) : tooltip$$1}; - function scatter() { - // background cliping rectangle - var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; - var container = setupContainer( selection, namespace, bgcpRect, backgroundFill ); - pointKeys = d3.keys(data); - valuesX = pointKeys.map(valueExtractorX); - valuesY = pointKeys.map(valueExtractorY); - valuesR = pointKeys.map(valueExtractorR); + function violin () { + // for convenience in handling orientation specific values + var horizontalQ = (orient == 'horizontal') ? true : false; - var numberOfObjects = pointKeys.length; - var extentX = [Math.min(...valuesX) - domainPaddingX, Math.max(...valuesX) + domainPaddingX]; - var extentY = [Math.min(...valuesY) - domainPaddingY, Math.max(...valuesY) + domainPaddingY]; - var extentR = [Math.min(...valuesR) - domainPaddingR, Math.max(...valuesR) + domainPaddingR]; + // background cliping rectangle + var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; + var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); - scaleX.domain(extentX).range([0, spaceX]); - scaleY.domain(extentY).range([spaceY, 0]); - scaleR.domain(extentR).range([minRadius, maxRadius]); + // if grouping is undefined sort violinKeys by sortingFunction + var ordered = (grouping == undefined) ? d3.keys(data).sort(sortingFunction) : grouping; - var points = container.selectAll('.'+objectClass); - points = points.data(pointKeys); - var pEnter = points.enter().append('circle') - .attr('class', objectClass) - .attr('cx', 0).attr('cy', spaceY).attr('r', 0); + // console.log(ordered) - var pExit = points.exit(); + violinKeys = utils.arr.flatten(ordered); - points = points.merge(pEnter); - points.each(function(key, i){ - var t = d3.select(this), - currentData = data[key], - x = valuesX[i], - y = valuesY[i], - r = valuesR[i], - fillColor = colorFunction$$1(key, currentData, i, 'fill'), - strokeColor = colorFunction$$1(key, currentData, i, 'stroke'); + var calcValues = neededViolinValues() + .horizontalQ(horizontalQ) + .quartileKeys(quartileKeys) + .violinPointsExtractor(violinPointsExtractor) + .violinPointValueExtractor(violinPointValueExtractor); - t.transition().duration(transitionDuration).ease(easeFunc) - .attr('cx', scaleX(x)) - .attr('cy', scaleY(y)) - .attr('r', scaleR(r)) - .attr('fill', fillColor) - .attr('stroke', strokeColor) - .attr('stroke-width', pointStrokeWidth); + // augment valus + violinKeys.map(function(vk, i){ calcValues(vk, data); }); - t.on('mouseover', function(d, i){ - points.style('opacity', 0.2); - t.style('opacity', 1); - t.transition().duration(transitionDuration/2).ease(easeFunc) - .attr('stroke-width', pointStrokeWidth*2) - .attr('r', scaleR(r) * 1.5); + var numberOfObjects = violinKeys.length; - }); - t.node().addEventListener('mouseout', function(){ - container.selectAll('.'+objectClass).style('opacity', 1); - t.transition().duration(transitionDuration/2).ease(easeFunc) - .attr('stroke-width', pointStrokeWidth) - .attr('r', scaleR(r)); + var min = [].concat(...violinKeys.map(function(k, i){return data[k].utils.math.quartiles[quartileKeys[0]]})); + var max = [].concat(...violinKeys.map(function(k, i){return data[k].utils.math.quartiles[quartileKeys[quartileKeys.length - 1]]})); + var extent = [Math.min(...min) - domainPadding, Math.max(...max) + domainPadding]; + // console.log(extent, violinValues, ordered) - }); + // set the scale + scale.domain(extent).range(horizontalQ ? [0,spaceY] : [0, spaceX]); + var space = horizontalQ ? spaceX : spaceY; + // calculate object size + objectSize = utils.math.calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ); + // calculate spacer size if needed + spacerSize = utils.math.calculateWidthOfSpacer(ordered, space, objectSize, numberOfObjects, objectSpacer, overflowQ); + // make the nested groups + var spacerFunction = groupingSpacer() + .horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects) + .objectClass(objectClass).objectSize(objectSize).spacerSize(spacerSize) + .transitionDuration(transitionDuration).easeFunc(easeFunc) + .namespace(namespace); - }); + // move stuff + spacerFunction(container, ordered, 0); + // console.log(violinKeys, ordered, container.selectAll('g:not(.to-remove).'+objectClass).nodes()) + // for color function + var parentIndexArray = []; + container.selectAll('g:not(.to-remove).'+objectClass) + .each(function(d, i){if (utils.arr.hasQ(violinKeys, d)){ parentIndexArray.push(Number(d3.select(this).attr('parent-index')));}}); + // update color function + colorFunction$$1 = colorFunction$$1.colorBy() == 'index' + ? colorFunction$$1.dataExtent([0, Math.max(...parentIndexArray)]) + : colorFunction$$1.dataExtent(extent); - pExit.transition().duration(transitionDuration).ease(easeFunc) - .attr('cx', 0).attr('cy', spaceY).attr('r', 0) - .remove(); + /* violiin specific needs */ + + + var frequencyMax = Math.max(...[].concat(...violinKeys.map(function(k, i){return d3.max(data[k].frequencies)}))); + var vScale = d3.scaleLinear().domain([0, frequencyMax]).range([0, objectSize / 2]); + + var lArea = d3.line() + .x(function(d, i){ return horizontalQ ? -vScale(d.x) : scale(d.x)}) + .y(function(d, i){ return horizontalQ ? scale(extent[1]) - scale(d.y) : -vScale(d.y)}) + .curve(d3.curveBasis); + var rArea = d3.line() + .x(function(d, i){ return horizontalQ ? vScale(d.x) : scale(d.x)}) + .y(function(d, i){ return horizontalQ ? scale(extent[1]) - scale(d.y) : vScale(d.y)}) + .curve(d3.curveBasis); + + + + + + + container.selectAll('g:not(.to-remove).'+objectClass).each(function(key, i){ + var t = d3.select(this), + currentData = data[key]; + // needed because bug in selecting .to-remove + if (!utils.arr.hasQ(violinKeys, key)) {return} + var + i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'), + fillColor = colorFunction$$1(key, currentData, i, 'fill'), // prevent duplicate computation + strokeColor = colorFunction$$1(key, currentData, i, 'stroke'), + area = utils.sel.safeSelect(t, 'g', 'area'), + la = utils.sel.safeSelect(area, 'path', 'left'), + ra = utils.sel.safeSelect(area, 'path', 'right'), + quarts = utils.sel.safeSelect(t, 'g', 'quarts'), + lq3 = utils.sel.safeSelect(quarts, 'line', 'q3'), + lq1 = utils.sel.safeSelect(quarts, 'line', 'q1'), + q3 = currentData.utils.math.quartiles[quartileKeys[3]], + q2 = currentData.utils.math.quartiles[quartileKeys[2]], + q1 = currentData.utils.math.quartiles[quartileKeys[1]]; + + t.attr('transform', horizontalQ ? 'translate('+objectSize / 2+',0)' : 'translate(0,'+objectSize / 2+')' ); + // draw curve + la.transition().duration(transitionDuration).attr('d', function(dd, ii){ return lArea(currentData.contour)}) + .attr('fill', fillColor) + .attr('stroke', strokeColor) + .attr('stroke-width', objectStrokeWidth); + + ra.transition().duration(transitionDuration).attr('d', function(dd, ii){ return rArea(currentData.contour)}) + .attr('fill', fillColor) + .attr('stroke', strokeColor) + .attr('stroke-width', objectStrokeWidth); + + area.node().addEventListener('mouseover', function(dd, ii){ + container.selectAll('g.'+objectClass).style('opacity', 0.2); + t.style('opacity', 1); + la.attr('stroke-width',objectStrokeWidth*2); + ra.attr('stroke-width',objectStrokeWidth*2); + }); + area.node().addEventListener('mouseout', function(dd, ii){ + container.selectAll('g.'+objectClass).style('opacity', 1); + la.attr('stroke-width',objectStrokeWidth); + ra.attr('stroke-width',objectStrokeWidth); + }); + + if (pointsQ) { + var ptsContainer = utils.sel.safeSelect(t, 'g', 'points'); + var pts = ptsContainer.selectAll('.point').data(currentData.pointKeys); + pts.on('mouseover', null); + + + var ptsExit = pts.exit().transition().ease(easeFunc).duration(transitionDuration) + .attr('r', 0) + .attr('cy', horizontalQ ? scale(extent[1]) - scale(q2) : vScale(0)) + .attr('cx', horizontalQ ? vScale(0) : scale(q2)).remove(); + + var ptsEnter = pts.enter().append('circle').attr('class', 'point').attr('r', 0) + .attr('cx', horizontalQ ? 0 : scale(q2)) + .attr('cy', horizontalQ ? scale(q2) : 0); + + pts = pts.merge(ptsEnter); + + // console.log(pointsTooltip.header()) + + var pTTips = tooltip().selection(pts).data(violinPointsExtractor(key, currentData)) + .header(pointsTooltip.header()) + .keys(pointsTooltip.keys()) + .values(pointsTooltip.values()); + + pTTips(); + + var pMin = d3.min(currentData.pointValues), pMax = d3.max(currentData.pointValues); + + + + pts.transition().duration(transitionDuration).ease(easeFunc).attr('r', pointRadius) + .attr('cy', function(pointKey, ii){ + var dd = currentData.pointValues[ii]; + if (horizontalQ) { return scale(extent[1]) - scale(dd) } + var j = utils.arr.whichBin(currentData.binned, dd); + var r = Math.random(); + var n = vScale(r * currentData.frequencies[j] * 0.5); + var k = Math.random() > 0.5 ? n : -n; + return k + }) + .attr('cx', function(pointKey, ii){ + var dd = currentData.pointValues[ii]; + if (horizontalQ) { + var j = utils.arr.whichBin(currentData.binned, dd); + var r = Math.random(); + var n = vScale(r * currentData.frequencies[j] * 0.5); + var k = Math.random() > 0.5 ? n : -n; + return k + } + return scale(dd) + }) + .attr('stroke', function(dd, ii) { var dd = currentData.pointValues[ii]; return pointColorFunc(dd, 'stroke', strokeColor, pMin, pMax) }) + .attr('fill' , function(dd, ii) { var dd = currentData.pointValues[ii]; return pointColorFunc(dd, 'fill' , strokeColor, pMin, pMax) }) + .attr('stroke-width', pointStrokeWidth); + + ptsContainer.selectAll('circle.point').on('mouseover', function(dd, ii){ + container.selectAll('g.'+objectClass).style('opacity', 0.2); + t.style('opacity', 1); + la.attr('stroke-width',objectStrokeWidth*2); + ra.attr('stroke-width',objectStrokeWidth*2); + + container.selectAll('.point').style('opacity', 0.2); + d3.select(this).style('opacity', 1).attr('r', pointRadius * 2).attr('stroke-width',pointStrokeWidth*2); + }); + ptsContainer.selectAll('circle.point').on('mouseout', function(dd, ii){ + var e = document.createEvent('SVGEvents'); + e.initEvent('mouseout',true,true); + area.node().dispatchEvent(e); + + container.selectAll('.point').style('opacity', 1); + d3.select(this).attr('stroke-width', pointStrokeWidth).attr('r', pointRadius); + }); + } + else { + cV.selectAll('.point') + .transition().duration(transitionDuration).ease(easeFunc) + .attr('r', 0) + .attr('cy', horizontalQ ? scale(extent[1]) - scale(q2) : vScale(0)) + .attr('cx', horizontalQ ? vScale(0) : scale(q2)) + .remove(); + } + + + }); - tooltip$$1.selection(points) - .data(data); + tooltip$$1.selection(container.selectAll('g:not(.to-remove).'+objectClass + ' .area')); + if (tooltip$$1.data() == undefined) {tooltip$$1.data(data);} tooltip$$1(); - } + if (tooltip$$1.values() == undefined) { + tooltip$$1.values([ + function(currentData, tooltipKey){ return currentData['utils.math.quartiles'][tooltipKey] }, + function(currentData, tooltipKey){ return currentData['utils.math.quartiles'][tooltipKey] }, + function(currentData, tooltipKey){ return currentData['utils.math.quartiles'][tooltipKey] }, + function(currentData, tooltipKey){ return currentData['utils.math.quartiles'][tooltipKey] }, + function(currentData, tooltipKey){ return currentData['utils.math.quartiles'][tooltipKey] } + ]); + } - return scatter + } + + return violin } -/******************************************************************************* -** ** -** ** -** PLOTZOOM ** -** ** -** ** -*******************************************************************************/ + + + /** - * Creates an plotZoom instance, which can handle drag and scroll events - * @constructor plotZoom - * @param {function} chart a function instance of one of the d3sm plots (e.g. bar, boxwhisker, bubbleHeatmap, violin, etc) - * @param {axis} xAxis the axis instance responsible for the x axis - * @param {axis} yAxis the axis instance responsible for the y axis - * @namespace plotZoom - * @returns {function} zoom - */ -function plotZoom( chart, xAxis, yAxis ) { +* Produces the function which manipulates the violin data to have values needed +* for rendering the violins as svg. +* @returns {function} calculateViolinValues +* @namespace neededViolinValues +*/ +function neededViolinValues() { var /** - * The event on which to fire - * (see {@link plotZoom#eventType}) - * @param {string} eventType which event it should handle. Currently supports scroll and wheel - * @memberof plotZoom# - * @property - */ - eventType, - /** - * A scaling factor for the wheel "speed" - * (see {@link plotZoom#wheelSpeed}) - * @param {number} [wheelSpeed=20] scales the wheel translation by wheelSpeed - * @memberof plotZoom# + * Whether or not the orientation of the violins are horizontal + * (see {@link violin#orient}) + * @param {Object} [horizontalQ=true] + * @memberof neededViolinValues# * @property */ - wheelSpeed = 20, + horizontalQ = true, /** - * The orientation in which to allow scrolling: 'horizontal', 'vertical', or '2D' - * (see {@link plotZoom#orient}) - * @param {string} [orient=chart.orient() || 'horizontal'] - * @memberof plotZoom# + * Keys to be put into the utils.math.quartiles if they need to be calculated. + * (see {@link violin#quartileKeys}) + * @param {Object} [quartileKeys=['Q0', 'Q1', 'Q2', 'Q3', 'Q4']] + * @memberof neededViolinValues# * @property */ - orient = (chart.orient == undefined) ? 'horizontal' : chart.orient(), + quartileKeys = ['Q0', 'Q1', 'Q2', 'Q3', 'Q4'], /** - * The max distance allowed to scroll in the x direction - * (see {@link plotZoom#xLock}) - * @param {number} [xLock=chart.spaceX()] ideally chart.overflowQ() == true and this value is the - * bounding rect across all elements in the chart minus the space in which to show. - * @memberof plotZoom# + * Function which given the key of the violin and that key's associated value + * returns the object consiting of the points of the violin + * (see {@link violin#violinPointsExtractor}) + * @param {Object} [violinPointsExtractor=function(violinKey, violinData) { return violinData.points }] + * @memberof neededViolinValues# * @property */ - xLock=chart.spaceX(), + violinPointsExtractor, /** - * The max distance allowed to scroll in the y direction - * (see {@link plotZoom#yLock}) - * @param {number} [yLock=chart.spaceY()] ideally chart.overflowQ() == true and this value is the - * bounding rect across all elements in the chart minus the space in which to show. - * @memberof plotZoom# + * Function which given the key of the current point and the object of points for the + * violin, returns the numerical value of the point + * (see {@link violin#violinPointValueExtractor}) + * @param {Object} [violinPointValueExtractor=function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value }] + * @memberof neededViolinValues# * @property */ - yLock=chart.spaceY(), - - chartSel = chart.selection(), - xAxisSel = xAxis.selection(), - yAxisSel = yAxis.selection(), - svg = d3.select(chartSel.thisSVG()); + violinPointValueExtractor; /** - * Gets or sets the event type in which to respond - * (see {@link plotZoom#eventType}) - * @param {string} [_=none] should be 'drag' or 'wheel' - * @returns {zoom | string} - * @memberof plotZoom + * Gets / sets the horizontalQ + * (see {@link violin#orient}) + * @param {boolean} [_=none] + * @returns {calculateViolinValues | boolean} + * @memberof calculateViolinValues * @property - * by default plotZoom=undefined + * + * by default horizontalQ = true */ - zoom.eventType = function(_) { return arguments.length ? (eventType = _, zoom) : eventType; }; + calculateViolinValues.horizontalQ = function(_) { return arguments.length ? (horizontalQ=_, calculateViolinValues) : horizontalQ }; /** - * Gets or sets the wheel speed in which to scale the wheel scroll transform - * (see {@link plotZoom#wheelSpeed}) - * @param {number} [_=none] - * @returns {zoom | number} - * @memberof plotZoom + * Gets / sets the quartileKeys + * (see {@link violin#quartileKeys}) + * @param {string[]} [_=none] + * @returns {calculateViolinValues | string[]} + * @memberof calculateViolinValues * @property - * by default wheelSpeed=20 + * + * by default quartileKeys = ["Q0","Q1","Q2","Q3","Q4"] */ - zoom.wheelSpeed = function(_) { return arguments.length ? (wheelSpeed = _, zoom) : wheelSpeed; }; + calculateViolinValues.quartileKeys = function(_) { return arguments.length ? (quartileKeys=_, calculateViolinValues) : quartileKeys }; /** - * Gets or sets the orientation in which items are manipulated - * (see {@link plotZoom#orient}) - * @param {string} [_=none] should be horizontal, vertical, or 2D - * @returns {zoom | string} - * @memberof plotZoom + * Gets / sets the violinPointsExtractor + * (see {@link violin#violinPointsExtractor}) + * @param {function} [_=none] + * @returns {calculateViolinValues | function} + * @memberof calculateViolinValues * @property - * by default orient=chart.orient() || 'horizontal' + * + * by default violinPointsExtractor = function(violinKey, violinData) { return violinData.points } */ - zoom.orient = function(_) { return arguments.length ? (orient = _, zoom) : orient; }; - + calculateViolinValues.violinPointsExtractor = function(_) { return arguments.length ? (violinPointsExtractor=_, calculateViolinValues) : violinPointsExtractor }; /** - * Gets or sets the max distance in which to scroll X - * (see {@link plotZoom#xLock}) - * @param {number} [_=none] should be a positive value - * @returns {zoom | number} - * @memberof plotZoom + * Gets / sets the violinPointValueExtractor + * (see {@link violin#violinPointValueExtractor}) + * @param {function} [_=none] + * @returns {calculateViolinValues | function} + * @memberof calculateViolinValues * @property - * by default xLock=chart.spaceX() + * + * by default violinPointValueExtractor = function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value } */ - zoom.xLock = function(_) { return arguments.length ? (xLock = _, zoom) : xLock; }; + calculateViolinValues.violinPointValueExtractor = function(_) { return arguments.length ? (violinPointValueExtractor=_, calculateViolinValues) : violinPointValueExtractor }; + + /** - * Gets or sets the max distance in which to scroll Y - * (see {@link plotZoom#yLock}) - * @param {number} [_=none] should be a positive value - * @returns {zoom | number} - * @memberof plotZoom - * @property - * by default yLock=chart.spaceY() - */ - zoom.yLock = function(_) { return arguments.length ? (yLock = _, zoom) : yLock; }; + * The function produced by neededViolinValues. + * + * Adds the data need to render the violin as an svg + * @param {string} violinKey the key of the current violin + * @param {object} data the object consisting of violinKey - values (violin data) pairs + * @returns {none} this function manipulates the passed data object adding the following keys + * + * data[violinKey].binned // the binned values of the points + * + * data[violinKey].frequencies // the list consisting of the length of each bin + * + * data[violinKey].contour // the points depicting the contour of 1/2 of the violin + * + * data[violinKey].utils.math.quartiles // the utils.math.quartiles of the points' values + * + * data[violinKey].pointKeys // the keys associated with the points + * + * data[violinKey].pointValues // the numerical values of the points + * + * @memberof neededViolinValues# + * @property + */ + function calculateViolinValues(violinKey, data) { + // data for the current violin + var violinData = data[violinKey]; + // the object of points + var violinPoints = violinPointsExtractor(violinKey, violinData); + // + var violinPointsKeys = d3.keys(violinPoints); + // the numerical values of those points + var violinPointsValues = violinPointsKeys.map(function(pk, i){return violinPointValueExtractor(pk, violinPoints)}); + // utils.math.quartiles of those points + var pointQuartiles = utils.math.quartiles(violinPointsValues, quartileKeys); - function setLocks() { - var chartObjSel = chartSel.select('.'+hypenate(chart.namespace(),'object-container')); - var chartObjTrans = getTranslation(chartObjSel.attr('transform')); - var cos = chartObjSel.attr('transform', 'translate(0,0)'); - xLock = chartSel.node().getBBox().width - chart.spaceX() * .9; - yLock = chartSel.node().getBBox().height - chart.spaceY() * .9; - cos.attr('transform', 'translate('+chartObjTrans[0]+','+chartObjTrans[1]+')'); - log('plotZoom', 'setLocks', {xLock:xLock, yLock:yLock}); + // binned points + var binned = d3.histogram()(violinPointsValues); + // length of bins + var frequencies = binned.map(bin=>bin.length); + // min and max countour points for nice drawings + var minContourPoint = horizontalQ ? {x: 0, y: d3.min(violinPointsValues)} : {x: d3.min(violinPointsValues), y: 0}; + var maxContourPoint = horizontalQ ? {x: 0, y: d3.max(violinPointsValues)} : {x: d3.max(violinPointsValues), y: 0}; + var violinContourPoints = binned.map(function(bin, i) { + return horizontalQ + ? {y: (bin.length) ? d3.median(bin): d3.median([bin.x0, bin.x1]), x: frequencies[i]} + : {x: (bin.length) ? d3.median(bin): d3.median([bin.x0, bin.x1]), y: frequencies[i]} + }); + // points along which to draw the violin shpe + violinContourPoints = [minContourPoint].concat(violinContourPoints).concat([maxContourPoint]); + + // set data + violinData.binned = binned; + violinData.frequencies = frequencies; + violinData.contour = violinContourPoints; + violinData.utils.math.quartiles = pointQuartiles; + violinData.pointKeys = violinPointsKeys; + violinData.pointValues = violinPointsValues; } + return calculateViolinValues +} - /** - * Sets the x and y locks (how far one can scroll) - * (see {@link plotZoom#xLock} and {@link plotZoom#yLock}) - * @function plotZoom.setLocks - * @returns {undefined} - * @memberof plotZoom - * @property - */ - zoom.setLocks = setLocks; +function upset ( selection ) { + var + data, + orient='horizontal', + spaceX, + spaceY, + overflowQ=false, + minObjectSize=20, + maxObjectSize=50, + circleStrokeWidth=2, + // colorFunction= + backgroundFill = 'transparent', + namespace='d3sm-upset', + objectClass = 'upset', - function zoom() { - setLocks(); + transitionDuration = 1000, + easeFunc = d3.easeExp, - var horizontalQ, verticalQ; - if (orient == '2D') {horizontalQ = true; verticalQ = true;} - if (orient == 'horizontal') {horizontalQ = true; verticalQ = false;} - if (orient == 'vertical') {verticalQ = true; horizontalQ = false;} + setKey = "set", + intersectionKey = "intersection", + elementsKey = "elements", - // capture transform event - var transform = d3.event.transform; + setExtractor = function(key, i) {return data[key][setKey]}, + intersectionExtractor = function(key, i) {return data[key][intersectionKey]}, + elementExtractor = function(key, i) {return data[key][elementsKey]}, - var chartBox = chartSel.node().getBBox(); - var xAxisBox = xAxisSel.node().getBBox(); - var yAxisBox = xAxisSel.node().getBBox(); + cellKeys, + setValues, + intersectionValues, + xObjectSpacer = 0.05, + yObjectSpacer = 0.05, + radius, - var chartWidth = chartBox.width - chartBox.x; - var chartHeight = chartBox.height - chartBox.y; - var xAxisWidth = xAxisBox.width;// - xAxisBox.x - var xAxisHeight = xAxisBox.height;// - xAxisBox.y - var yAxisWidth = yAxisBox.width;// - yAxisBox.x - var yAxisHeight = yAxisBox.height;// -yAxisBox.y + // listDelim = ';' - // enable wheel event - if (eventType == "wheel") { - var e = d3.event; - // prevent page scrolling - e.preventDefault(); - // event delta is very very slow, so speed it up with wheel speed - var w = d3.event.deltaY * wheelSpeed; - var shiftQ = d3.event.shiftKey; + yObjectSize, + ySpacerSize, + xObjectSize, + xSpacerSize, - // d3 has no way to make custom transform, so make an object and add - // the necessary functions to make wheel event compatible with drag events - if (orient == '2D') { - transform = shiftQ ? {k: 1, x: w, y: 0} : {k: 1, x: 0, y: w}; - } else { - transform = horizontalQ ? {k: 1, x: w, y: 0} : {k: 1, x: 0, y: w}; - } - // the * -1 inverts the direction - transform.applyX = function(x) { return x * this.k + this.x * -1; }; - transform.applyY = function(y) { return y * this.k + this.y * -1; }; - } + setKeySortingFunction = function(a, b) { return setValues.indexOf(setExtractor(a)) - setValues.indexOf(setExtractor(b)) }, + intersectionKeySortingFunction = function(a, b) { return intersectionValues.indexOf(intersectionExtractor(a)) - intersectionValues.indexOf(intersectionExtractor(b)) }; + upset.selection = function(_) { return arguments.length ? (selection = _, upset) : selection; }; + upset.data = function(_) { return arguments.length ? (data = _, upset) : data; }; + upset.orient = function(_) { return arguments.length ? (orient = _, upset) : orient; }; + upset.spaceX = function(_) { return arguments.length ? (spaceX = _, upset) : spaceX; }; + upset.spaceY = function(_) { return arguments.length ? (spaceY = _, upset) : spaceY; }; + upset.overflowQ = function(_) { return arguments.length ? (overflowQ = _, upset) : overflowQ; }; + upset.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, upset) : minObjectSize; }; + upset.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, upset) : maxObjectSize; }; + upset.circleStrokeWidth = function(_) { return arguments.length ? (circleStrokeWidth = _, upset) : circleStrokeWidth; }; + upset.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, upset) : backgroundFill; }; + upset.namespace = function(_) { return arguments.length ? (namespace = _, upset) : namespace; }; + upset.objectClass = function(_) { return arguments.length ? (objectClass = _, upset) : objectClass; }; + upset.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, upset) : transitionDuration; }; + upset.easeFunc = function(_) { return arguments.length ? (easeFunc = _, upset) : easeFunc; }; + upset.cellKeys = function(_) { return arguments.length ? (cellKeys = _, upset) : cellKeys; }; + upset.setValues = function(_) { return arguments.length ? (setValues = _, upset) : setValues; }; + upset.intersectionValues = function(_) { return arguments.length ? (intersectionValues = _, upset) : intersectionValues; }; + upset.xObjectSpacer = function(_) { return arguments.length ? (xObjectSpacer = _, upset) : xObjectSpacer; }; + upset.yObjectSpacer = function(_) { return arguments.length ? (yObjectSpacer = _, upset) : yObjectSpacer; }; + upset.radius = function(_) { return arguments.length ? (radius = _, upset) : radius; }; + upset.setExtractor = function(_) { return arguments.length ? (setExtractor = _, upset) : setExtractor; }; + upset.intersectionExtractor = function(_) { return arguments.length ? (intersectionExtractor = _, upset) : intersectionExtractor; }; + upset.elementExtractor = function(_) { return arguments.length ? (elementExtractor = _, upset) : elementExtractor; }; + upset.setKeySortingFunction = function(_) { return arguments.length ? (setKeySortingFunction = _, upset) : setKeySortingFunction; }; + upset.intersectionKeySortingFunction = function(_) { return arguments.length ? (intersectionKeySortingFunction = _, upset) : intersectionKeySortingFunction; }; - var chartObjSel = chartSel.select('.'+hypenate(chart.namespace(),'object-container')); - var xAxisObjSel = xAxisSel.select('.'+hypenate(xAxis.namespace(),'object-container')); - var yAxisObjSel = yAxisSel.select('.'+hypenate(yAxis.namespace(),'object-container')); + upset.yObjectSize = function(_) { return arguments.length ? (yObjectSize = _, upset) : yObjectSize; }; + upset.ySpacerSize = function(_) { return arguments.length ? (ySpacerSize = _, upset) : ySpacerSize; }; + upset.xObjectSize = function(_) { return arguments.length ? (xObjectSize = _, upset) : xObjectSize; }; + upset.xSpacerSize = function(_) { return arguments.length ? (xSpacerSize = _, upset) : xSpacerSize; }; - // xLock = chartSel.node().getBBox().width - chart.spaceX() - chartSel.node().getBBox().x - // yLock = chartSel.node().getBBox().height - chart.spaceY() - // console.table({'xLock':xLock, "yLock":yLock}) - // bhm.selection().node().getBBox().width - bhm.spaceX() + function upset() { + // for convenience in handling orientation specific values + var horizontalQ = (orient == 'horizontal') ? true : false; + var verticalQ = !horizontalQ; + // background cliping rectangle + var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; + var container = utils$1.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); - var chartObjTrans = getTranslation(chartObjSel.attr('transform')); - var xAxisObjTrans = getTranslation(xAxisObjSel.attr('transform')); - var yAxisObjTrans = getTranslation(yAxisObjSel.attr('transform')); + cellKeys = d3.keys(data); + setValues = utils$1.arr.unique(cellKeys.map(setExtractor)).sort(); + intersectionValues = utils$1.arr.unique(cellKeys.map(intersectionExtractor)).sort().sort(function(a, b){ + return a.split(';').length - b.split(';').length + }); - var x = horizontalQ ? transform.applyX(chartObjTrans[0]) : 0; - if (horizontalQ) {x = x < -xLock ? (transform.x = 0, -xLock) : (transform.x = 0, Math.min(x, 0)); } + if (!horizontalQ) { + cellKeys.sort(function(a, b){ return setKeySortingFunction(a, b) || intersectionKeySortingFunction(a, b) }); + } else { + cellKeys.sort(function(a, b){ return intersectionKeySortingFunction(a, b) || setKeySortingFunction(a, b) }); + } - var y = verticalQ ? transform.applyY(chartObjTrans[1]) : 0; - if (verticalQ) {y = y < -yLock ? (transform.y = 0, -yLock): (transform.y = 0, Math.min(y, 0));} - chartObjSel.attr('transform', 'translate('+x+','+y+')'); - if (horizontalQ) { xAxisObjSel.attr('transform', 'translate('+x+','+0+')'); } - if (verticalQ) { yAxisObjSel.attr('transform', 'translate('+0+','+y+')'); } - // var lasso = svg.select(".lasso-container") - // if (!lasso.empty()) { - // lasso.attr('transform', 'translate('+x+','+y+')') - // } - } + var + xValues = horizontalQ ? intersectionValues : setValues, + yValues = horizontalQ ? setValues : intersectionValues, + xDim = horizontalQ ? xValues.length : yValues.length, + yDim = horizontalQ ? yValues.length : xValues.length; - zoom.reset = function() { - - var chartObjSel = chartSel.select('.'+hypenate(chart.namespace(),'object-container')); - var xAxisObjSel = xAxisSel.select('.'+hypenate(xAxis.namespace(),'object-container')); - var yAxisObjSel = yAxisSel.select('.'+hypenate(yAxis.namespace(),'object-container')); - chartObjSel.attr('transform', 'translate('+0+','+0+')'); - xAxisObjSel.attr('transform', 'translate('+0+','+0+')'); - yAxisObjSel.attr('transform', 'translate('+0+','+0+')'); - }; + // console.utils.con.log(xValues, yValues) - return zoom -} -/******************************************************************************* + xObjectSize = utils$1.math.calculateWidthOfObject(spaceX, xDim, minObjectSize, maxObjectSize, xObjectSpacer, overflowQ); + yObjectSize = utils$1.math.calculateWidthOfObject(spaceY, yDim, minObjectSize, maxObjectSize, yObjectSpacer, overflowQ); + xSpacerSize = utils$1.math.calculateWidthOfSpacer(xValues, spaceX, xObjectSize, xDim, xObjectSpacer, overflowQ); + ySpacerSize = utils$1.math.calculateWidthOfSpacer(yValues, spaceY, yObjectSize, yDim, yObjectSpacer, overflowQ); -** ** -** ** -** PLOTZOOM ** -** ** -** ** -*******************************************************************************/ -/** - * Creates an plotZoom instance, which can handle drag and scroll events - * @constructor plotZoom - * @param {function} chart a function instance of one of the d3sm plots (e.g. bar, boxwhisker, bubbleHeatmap, violin, etc) - * @param {axis} xAxis the axis instance responsible for the x axis - * @param {axis} yAxis the axis instance responsible for the y axis - * @namespace plotZoom - * @returns {function} zoom - */ -function multiPlotZoom( chart ) { - var - /** - * The event on which to fire - * (see {@link plotZoom#eventType}) - * @param {string} eventType which event it should handle. Currently supports scroll and wheel - * @memberof plotZoom# - * @property - */ - eventType, - /** - * A scaling factor for the wheel "speed" - * (see {@link plotZoom#wheelSpeed}) - * @param {number} [wheelSpeed=20] scales the wheel translation by wheelSpeed - * @memberof plotZoom# - * @property - */ - wheelSpeed = 20, - /** - * The orientation in which to allow scrolling: 'horizontal', 'vertical', or '2D' - * (see {@link plotZoom#orient}) - * @param {string} [orient=chart.orient() || 'horizontal'] - * @memberof plotZoom# - * @property - */ - orient = (chart.orient == undefined) ? 'horizontal' : chart.orient(), - /** - * The max distance allowed to scroll in the x direction - * (see {@link plotZoom#xLock}) - * @param {number} [xLock=chart.spaceX()] ideally chart.overflowQ() == true and this value is the - * bounding rect across all elements in the chart minus the space in which to show. - * @memberof plotZoom# - * @property - */ - xLock=chart.spaceX(), - /** - * The max distance allowed to scroll in the y direction - * (see {@link plotZoom#yLock}) - * @param {number} [yLock=chart.spaceY()] ideally chart.overflowQ() == true and this value is the - * bounding rect across all elements in the chart minus the space in which to show. - * @memberof plotZoom# - * @property - */ - yLock=chart.spaceY(), + var ySpacer = groupingSpacer() + .horizontalQ(false) + .moveby('category').numberOfObjects(yDim) + .objectSize(yObjectSize).spacerSize(ySpacerSize) + .transitionDuration(transitionDuration).easeFunc(easeFunc); - chartSel = chart.selection(), + var xSpacer = groupingSpacer() + .horizontalQ(true) + .moveby('category').numberOfObjects(xDim) + .objectClass(objectClass) + .objectSize(xObjectSize).spacerSize(xSpacerSize) + .transitionDuration(transitionDuration).easeFunc(easeFunc); - svg = d3.select(chartSel.thisSVG()), - xComponents = [], - yComponents = []; + if (verticalQ) { + xSpacer.objectClass(objectClass); + ySpacer.namespace('across').objectClass(utils$1.str.hypenate(objectClass, 'across')); - /** - * Gets or sets the event type in which to respond - * (see {@link plotZoom#eventType}) - * @param {string} [_=none] should be 'drag' or 'wheel' - * @returns {zoom | string} - * @memberof plotZoom - * @property - * by default plotZoom=undefined - */ - zoom.eventType = function(_) { return arguments.length ? (eventType = _, zoom) : eventType; }; - /** - * Gets or sets the wheel speed in which to scale the wheel scroll transform - * (see {@link plotZoom#wheelSpeed}) - * @param {number} [_=none] - * @returns {zoom | number} - * @memberof plotZoom - * @property - * by default wheelSpeed=20 - */ - zoom.wheelSpeed = function(_) { return arguments.length ? (wheelSpeed = _, zoom) : wheelSpeed; }; - /** - * Gets or sets the orientation in which items are manipulated - * (see {@link plotZoom#orient}) - * @param {string} [_=none] should be horizontal, vertical, or 2D - * @returns {zoom | string} - * @memberof plotZoom - * @property - * by default orient=chart.orient() || 'horizontal' - */ - zoom.orient = function(_) { return arguments.length ? (orient = _, zoom) : orient; }; + ySpacer(container, yValues, 0); + container.selectAll('g.'+utils$1.str.hypenate(objectClass, 'across')) + .each(function(d, i){ xSpacer(d3.select(this), xValues, 0); }); + } else { + xSpacer.namespace('across').objectClass(utils$1.str.hypenate(objectClass, 'across')); + ySpacer.objectClass(objectClass); - /** - * Gets or sets the max distance in which to scroll X - * (see {@link plotZoom#xLock}) - * @param {number} [_=none] should be a positive value - * @returns {zoom | number} - * @memberof plotZoom - * @property - * by default xLock=chart.spaceX() - */ - zoom.xLock = function(_) { return arguments.length ? (xLock = _, zoom) : xLock; }; - /** - * Gets or sets the max distance in which to scroll Y - * (see {@link plotZoom#yLock}) - * @param {number} [_=none] should be a positive value - * @returns {zoom | number} - * @memberof plotZoom - * @property - * by default yLock=chart.spaceY() - */ - zoom.yLock = function(_) { return arguments.length ? (yLock = _, zoom) : yLock; }; + xSpacer(container, xValues, 0); + container.selectAll('g.'+utils$1.str.hypenate(objectClass, 'across')) + .each(function(d, i){ ySpacer(d3.select(this), yValues, 0); }); + } - zoom.xComponents = function(_) { return arguments.length ? (xComponents = _, zoom) : xComponents; }; - zoom.yComponents = function(_) { return arguments.length ? (yComponents = _, zoom) : yComponents; }; + var cells = container.selectAll('g:not(.to-remove).'+objectClass); + var lookup = {}; + cellKeys.map(function(k, i){ + lookup[setExtractor(k)+'::'+intersectionExtractor(k)] = k; + }); - function setLocks() { - var chartObjSel = chartSel.select('.'+hypenate(chart.namespace(),'object-container')); - var chartObjTrans = getTranslation(chartObjSel.attr('transform')); - var cos = chartObjSel.attr('transform', 'translate(0,0)'); + // var positionedCellKeys = [] + // for (var i = 0; i < setValues.length; i++) { + // for (var j = 0; j < intersectionValues.length; j++) { + // var lookupValue = lookup[setValues[j]+"::"+intersectionValues[i]] + // if (lookupValue == undefined) { + // positionedCellKeys.push(undefined) + // } else { + // positionedCellKeys.push(lookupValue) + // } + // console.utils.con.log(i, j, lookupValue) + // } + // } + // + // // console.utils.con.log(positionedCellKeys) + // + // cells.data(positionedCellKeys); - xLock = chartSel.node().getBBox().width - chart.spaceX();// * .9 - yLock = chartSel.node().getBBox().height - chart.spaceY();// * .9 - cos.attr('transform', 'translate('+chartObjTrans[0]+','+chartObjTrans[1]+')'); - log('plotZoom', 'setLocks', {xLock:xLock, yLock:yLock}); - } + cells.data(cellKeys); - /** - * Sets the x and y locks (how far one can scroll) - * (see {@link plotZoom#xLock} and {@link plotZoom#yLock}) - * @function plotZoom.setLocks - * @returns {undefined} - * @memberof plotZoom - * @property - */ - zoom.setLocks = setLocks; + cells.each(function(key, i) { + var t = d3.select(this); + if (key == undefined) {return } + var + currentData = data[key]; + // console.utils.con.log(key, currentData) + var + set = setExtractor(key, i), + intersection = intersectionExtractor(key, i); - function zoom() { - setLocks(); + // console.utils.con.log(set, intersection) - var - xComponentsSel = xComponents.map(function(d, i){return d.selection()}), - yComponentsSel = yComponents.map(function(d, i){return d.selection()}); + t.classed(intersection, true); + t.classed(set, true); - var horizontalQ, verticalQ; - if (orient == '2D') {horizontalQ = true; verticalQ = true;} - if (orient == 'horizontal') {horizontalQ = true; verticalQ = false;} - if (orient == 'vertical') {verticalQ = true; horizontalQ = false;} + var c = utils$1.sel.safeSelect(t, 'circle', utils$1.str.hypenate(objectClass,'circle')); + c.attr('cx', xObjectSize / 2) + .attr('cy', yObjectSize / 2 ) + .attr('r', radius == undefined ? Math.min(xObjectSize, yObjectSize) / 2 : radius) + .attr('fill', intersection.includes(set) ? "black": 'rgb(233,233,233)') + .attr('stroke', "black") + .attr("in-intersection", intersection.includes(set)); + }); - // capture transform event - var transform = d3.event.transform; - var chartBox = chartSel.node().getBBox(); - var xComponentsBox = xComponentsSel.map(function(d, i){return d.node().getBBox()}); - var yComponentsBox = xComponentsSel.map(function(d, i){return d.node().getBBox()}); + } - var chartWidth = chartBox.width - chartBox.x; - var chartHeight = chartBox.height - chartBox.y; + function intersectionTotals() { + var totals = {}; + // intersectionValues.sort(function(a, b){ return intersectionKeySortingFunction(a, b) }) - // enable wheel event - if (eventType == "wheel") { - var e = d3.event; - // prevent page scrolling - e.preventDefault(); - // event delta is very very slow, so speed it up with wheel speed - var w = d3.event.deltaY * wheelSpeed; - var shiftQ = d3.event.shiftKey; + intersectionValues.map(function(k, i){ totals[k] = {'total': 0}; }); + cellKeys.map(function(k, i){ + var e = elementExtractor(k, i); + if (totals[intersectionExtractor(k, i)]['total'] == 0) { + if (Array.isArray(e)) { + totals[intersectionExtractor(k, i)]['total']+= e.length; + totals[intersectionExtractor(k, i)]['values'] = e; + } else { + totals[intersectionExtractor(k, i)]['total']+= e; + } - // d3 has no way to make custom transform, so make an object and add - // the necessary functions to make wheel event compatible with drag events - if (orient == '2D') { - transform = shiftQ ? {k: 1, x: w, y: 0} : {k: 1, x: 0, y: w}; - } else { - transform = horizontalQ ? {k: 1, x: w, y: 0} : {k: 1, x: 0, y: w}; } - // * -1 is invert - transform.applyX = function(x) { return x * this.k + this.x *-1; }; - transform.applyY = function(y) { return y * this.k + this.y *-1; }; - } - + }); + return totals + } + function setTotals(){ + var totals = {}; + // intersectionValues.sort(function(a, b){ return intersectionKeySortingFunction(a, b) }) - var chartObjSel = chartSel.select('.'+hypenate(chart.namespace(),'object-container')); - var xComponentObjSel = xComponentsSel.map(function(d, i){ - return d.select('.'+hypenate(xComponents[i].namespace(),'object-container')) - }); - var yComponentObjSel = yComponentsSel.map(function(d, i){ - return d.select('.'+hypenate(yComponents[i].namespace(),'object-container')) - }); + setValues.map(function(k, i){ totals[k] = {'total': 0}; }); - var chartObjTrans = getTranslation(chartObjSel.attr('transform')); - var xComponentsObjTrans = xComponentObjSel.map(function(d, i){ - return getTranslation(d.attr('transform')) - }); - var yComponentsObjTrans = yComponentObjSel.map(function(d, i){ - return getTranslation(d.attr('transform')) + cellKeys.map(function(k, i){ + var e = elementExtractor(k, i); + if (Array.isArray(e)) { + totals[setExtractor(k, i)]['total']+= e.length; + } else { + totals[setExtractor(k, i)]['total']+= e; + } }); - - var x = horizontalQ ? transform.applyX(chartObjTrans[0]) : 0; - if (horizontalQ) {x = x < -xLock ? (transform.x = 0, -xLock) : (transform.x = 0, Math.min(x, 0)); } - - var y = verticalQ ? transform.applyY(chartObjTrans[1]) : 0; - if (verticalQ) {y = y < -yLock ? (transform.y = 0, -yLock): (transform.y = 0, Math.min(y, 0));} - - chartObjSel.attr('transform', 'translate('+x+','+y+')'); - if (horizontalQ) { - // xAxisObjSel.attr('transform', 'translate('+x+','+0+')') - xComponentObjSel.map(function(d, i){ d.attr('transform', 'translate('+x+','+0+')'); }); - } - if (verticalQ) { - // yAxisObjSel.attr('transform', 'translate('+0+','+y+')') - yComponentObjSel.map(function(d, i){ d.attr('transform', 'translate('+0+','+y+')'); }); - - } - - + return totals } - zoom.reset = function() { - - var chartObjSel = chartSel.select('.'+hypenate(chart.namespace(),'object-container')); - var xAxisObjSel = xAxisSel.select('.'+hypenate(xAxis.namespace(),'object-container')); - var yAxisObjSel = yAxisSel.select('.'+hypenate(yAxis.namespace(),'object-container')); - chartObjSel.attr('transform', 'translate('+0+','+0+')'); - xAxisObjSel.attr('transform', 'translate('+0+','+0+')'); - yAxisObjSel.attr('transform', 'translate('+0+','+0+')'); - }; + upset.intersectionTotals = intersectionTotals; + upset.setTotals = setTotals; - return zoom + return upset } -/******************************************************************************* -** ** -** ** -** VIOLIN ** -** ** -** ** -*******************************************************************************/ +let charts = { + scatter, bar, bubble: bubbleHeatmap, heatmap, violin, neededViolinValues, upset +}; + +function categoricLegend( selection ) { + var + categories, -/** - * Creates a violin - * - * {@link https://sumneuron.gitlab.io/d3sm/demos/basic-violins/index.html Demo} - * @constructor violin - * @param {d3.selection} selection - * @namespace violin - * @returns {function} violin - */ -function violin( selection ) { - var /** - * Data to plot. Assumed to be a object, where each key corresponds to a violin - * (see {@link violin#data}) + * Data to plot. Assumed to be a object, where each key corresponds to a legend + * (see {@link legend#data}) * @param {Object} [data=undefined] - * @memberof violin# + * @memberof legend# * @property */ data, /** * Which direction to render the bars in - * (see {@link violin#orient}) + * (see {@link legend#orient}) * @param {number} [orient='horizontal'] - * @memberof violin# + * @memberof legend# * @property */ orient='horizontal', /** - * Amount of horizontal space (in pixels) avaible to render the violin in - * (see {@link violin#spaceX}) + * Amount of horizontal space (in pixels) avaible to render the legend in + * (see {@link legend#spaceX}) * @param {number} [spaceX=undefined] - * @memberof violin# + * @memberof legend# * @property */ spaceX, /** - * Amount of vertical space (in pixels) avaible to render the violin in - * (see {@link violin.spaceY}) + * Amount of vertical space (in pixels) avaible to render the legend in + * (see {@link legend.spaceY}) * @param {number} [spaceY=undefined] - * @memberof violin# + * @memberof legend# * @property */ spaceY, + /** - * Whether or not to allow violin to render elements pass the main spatial dimension - * given the orientation (see {@link violin#orient}), where {@link violin#orient}="horizontal" - * the main dimension is {@link violin#spaceX} and where {@link violin#orient}="vertical" - * the main dimension is {@link violin#spaceY} + * Whether or not to allow legend to render elements pass the main spatial dimension + * given the orientation (see {@link legend#orient}), where {@link legend#orient}="horizontal" + * the main dimension is {@link legend#spaceX} and where {@link legend#orient}="vertical" + * the main dimension is {@link legend#spaceY} * @param {boolean} [overflowQ=false] - * @memberof violin# - * @property - */ - overflowQ = true, - /** - * Whether or not to display points inside the points - * @param {boolean} [pointsQ=false] - * @memberof violin# + * @memberof legend# * @property */ - pointsQ = true, + overflowQ = false, + /** * An array - putatively of other arrays - depicting how bars should be arranged * @param {Array[]} [grouping=undefined] - * @memberof violin# + * @memberof legend# * @property */ grouping, + /** - * How to get the value of the violin + * How to get the value of the legend * @param {function} [valueExtractor=function(key, index) { return data[key] }] - * @memberof violin# + * @memberof legend# * @property */ - valueExtractor = function(key, index) {return data[key] }, + valueExtractor = function(key, index) { return data[key] }, /** - * How to sort the bars - if {@link violin#grouping} is not provided. + * How to sort the bars - if {@link bar#grouping} is not provided. * @param {function} [sortingFunction=function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}] - * @memberof violin# - * @property - */ - sortingFunction = function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}, - - /** - * The scale for which violin values should be transformed by - * @param {d3.scale} [scale=d3.scaleLinear] - * @memberof violin# - * @property - */ - scale = d3.scaleLinear(), - /** - * The padding for the domain of the scale (see {@link violin#scale}) - * @param {number} [domainPadding=0.5] - * @memberof violin# + * @memberof bar# * @property */ - domainPadding = 0.5, + sortingFunction = function(keyA, keyB) {return d3.ascending(keyA, keyB)}, /** * Default space for the spacer (percentage) of main dimension given the orientation - * (see {@link violin#orient}), where {@link violin#orient}="horizontal" - * the main dimension is {@link violin#spaceX} and where {@link violin#orient}="vertical" - * the main dimension is {@link violin#spaceY} between bars + * (see {@link legend#orient}), where {@link legend#orient}="horizontal" + * the main dimension is {@link legend#spaceX} and where {@link legend#orient}="vertical" + * the main dimension is {@link legend#spaceY} between bars * @param {number} [objectSpacer=0.05] - * @memberof violin# + * @memberof legend# * @property */ objectSpacer = 0.05, /** * The minimum size that an object can be * @param {number} [minObjectSize=50] - * @memberof violin# + * @memberof legend# * @property */ - minObjectSize = 50, + minObjectSize = 10, /** * The maximum size that an object can be * @param {number} [maxObjectSize=100] - * @memberof violin# + * @memberof legend# * @property */ maxObjectSize = 100, @@ -7252,2947 +7024,1410 @@ function violin( selection ) { /** * The stroke width of the bars * @param {number} [barStrokeWidth=2] - * @memberof violin# + * @memberof legend# * @property */ - objectStrokeWidth = 2, + bubbleStrokeWidth = 2, /** * Instance of ColorFunction * @param {function} [colorFunction = colorFunction()] - * @memberof violin# + * @memberof legend# * @property */ colorFunction$$1 = colorFunction(), - /** - * Instance of ColorFunction modified by a scale for the points - * @param {function} [pointColorFunc = colorFunction()] - * @memberof violin# - * @property - */ - pointColorFunc = function (d, type, base, min, max) { - var minMaxHexScale = d3.scaleLinear().domain([min, max]).range([-0.25, 0.05]); - var scaledColor = modifyHexidecimalColorLuminance(base.replace('#', ''), minMaxHexScale(d)); - var mod = type == "stroke" ? 0 : 0.25; - return modifyHexidecimalColorLuminance(scaledColor.replace('#', ''), mod) - }, - - /** - * The radius of a point - * @param {number} [pointRadius=3] - * @memberof violin# - * @property - */ - pointRadius = 3, - /** - * The stroke width of the oints - * @param {number} [pointStrokeWidth=2] - * @memberof violin# - * @property - */ - pointStrokeWidth = 2, /** * Color of the background * @param {string} [backgroundFill="transparent"] - * @memberof violin# + * @memberof legend# * @property */ backgroundFill = 'transparent', /** - * Namespace for all items made by this instance of violin - * @param {string} [namespace="d3sm-violin"] - * @memberof violin# + * Namespace for all items made by this instance of legend + * @param {string} [namespace="d3sm-legend"] + * @memberof legend# * @property */ - namespace = 'd3sm-violin', + namespace = 'd3sm-legend', /** - * Class name for violin container ( element) - * @param {string} [objectClass="violin"] - * @memberof violin# + * Class name for legend container ( element) + * @param {string} [objectClass="legend"] + * @memberof legend# * @property */ - objectClass = 'violin', + objectClass = 'legend', + /** * Duration of all transitions of this element * @param {number} [transitionDuration=1000] - * @memberof violin# + * @memberof legend# * @property */ transitionDuration = 1000, /** * Easing function for transitions * @param {d3.ease} [easeFunc=d3.easeExp] - * @memberof violin# + * @memberof legend# * @property */ - easeFunc = d3.easeExp, + easeFunc = d3.easeExp; - /** - * The keys corresponding to each quartile - * @param {string[]} [quartileKeys=["Q0", "Q1", "Q2", "Q3", "Q4"]] - * @memberof violin# - * @property - */ - quartileKeys = ["Q0", "Q1", "Q2", "Q3", "Q4"], + + legend.categories = function(_) { return arguments.length ? (categories=_, legend) : categories }; /** - * The keys of the bars - * @param {string[]} [violinKeys=undefined] - * @memberof violin# - * @property - */ - violinKeys, - /** - * The values of the bars - * @param {number[]} [violinValues=undefined] - * @memberof violin# - * @property - */ - violinValues, + * Gets or sets the selection in which items are manipulated + * @param {d3.selection} [_=none] + * @returns {legend | d3.selection} + * @memberof legend + * @property + * by default selection = selection + */ + legend.selection = function(_) { return arguments.length ? (selection = _, legend) : selection; }; /** - * The objectSize (actual width) used by the bars - * @param {number} [objectSize=undefined] - * @memberof violin# - * @property - */ - objectSize, + * Gets or sets the data + * (see {@link legend#data}) + * @param {number} [_=none] + * @returns {legend | object} + * @memberof legend + * @property + */ + legend.data = function(_) { return arguments.length ? (data = _, legend) : data; }; /** - * The spacerSize (actual width) used by the spacers between the bars - * @param {number} [spacerSize=undefined] - * @memberof violin# - * @property - */ - spacerSize, - - /** - * Instance of Tooltip - * @param {function} [tooltip=tooltip()] - * @memberof violin# - * @property - */ - tooltip$$1 = tooltip().keys([quartileKeys[4], quartileKeys[3], quartileKeys[2], quartileKeys[1], quartileKeys[0]]), - pointsTooltip = tooltip(), - // pointKeyExtractor = function(violinKey, violinData, violinValues) {return d3.keys(violinValues[violinKey].values)}, - // pointValueExtractor = function(pointKey, violinKey, violinData, violinValues) {return violinValues[violinKey].values[pointKey]}, - - - /** - * Function which given the key of the violin and that key's associated value - * returns the object consiting of the points of the violin - * (see {@link violin#violinPointsExtractor}) - * @param {Object} [violinPointsExtractor=function(violinKey, violinData) { return violinData.points }] - * @memberof violin# - * @property - */ - violinPointsExtractor = function (violinKey, violinData) {return violinData.points }, - /** - * Function which given the key of the current point and the object of points for the - * violin, returns the numerical value of the point - * (see {@link violin#violinPointValueExtractor}) - * @param {Object} [violinPointValueExtractor=function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value }] - * @memberof violin# - * @property - */ - violinPointValueExtractor = function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value }; - - - /** - * Gets or sets the violinPointsExtractor - * @param {function} [_=none] - * @returns {violin | function} - * @memberof violin - * @property - * by default violinPointsExtractor = function(violinKey, violinData) { return violinData.points } - */ - violin.violinPointsExtractor = function(_) { return arguments.length ? (violinPointsExtractor = _, violin) : violinPointsExtractor; }; - /** - * Gets or sets the violinPointValueExtractor - * @param {function} [_=none] - * @returns {violin | function} - * @memberof violin - * @property - * by default violinPointsExtractor = function(pointKey, violinKey, violinData, violinValues) {return violinValues[violinKey].values[pointKey]} - */ - violin.violinPointValueExtractor = function(_) { return arguments.length ? (violinPointValueExtractor = _, violin) : violinPointValueExtractor; }; - - - /** - * Gets or sets the selection in which items are manipulated - * @param {d3.selection} [_=none] - * @returns {violin | d3.selection} - * @memberof violin - * @property - * by default selection = selection - */ - violin.selection = function(_) { return arguments.length ? (selection = _, violin) : selection; }; - /** - * Gets or sets the data - * (see {@link violin#data}) - * @param {number} [_=none] - * @returns {violin | object} - * @memberof violin - * @property - */ - violin.data = function(_) { return arguments.length ? (data = _, violin) : data; }; - /** - * Gets or sets the orient of the boxes - * (see {@link violin#orient}) + * Gets or sets the orient of the bars + * (see {@link legend#orient}) * @param {number} [_=none] - * @returns {violin | object} - * @memberof violin + * @returns {legend | object} + * @memberof legend * @property */ - violin.orient = function(_) { return arguments.length ? (orient = _, violin) : orient; }; + legend.orient = function(_) { return arguments.length ? (orient = _, legend) : orient; }; /** * Gets or sets the amount of horizontal space in which items are manipulated - * (see {@link violin#spaceX}) + * (see {@link legend#spaceX}) * @param {number} [_=none] should be a number > 0 - * @returns {violin | number} - * @memberof violin + * @returns {legend | number} + * @memberof legend * @property * by default spaceX = undefined */ - violin.spaceX = function(_) { return arguments.length ? (spaceX = _, violin) : spaceX; }; + legend.spaceX = function(_) { return arguments.length ? (spaceX = _, legend) : spaceX; }; /** * Gets or sets the amount of vertical space in which items are manipulated - * (see {@link violin#spaceY}) + * (see {@link legend#spaceY}) * @param {number} [_=none] should be a number > 0 - * @returns {violin | number} - * @memberof violin + * @returns {legend | number} + * @memberof legend * @property * by default spaceY = undefined */ - violin.spaceY = function(_) { return arguments.length ? (spaceY = _, violin) : spaceY; }; - + legend.spaceY = function(_) { return arguments.length ? (spaceY = _, legend) : spaceY; }; /** - * Gets / sets whether or not violin is allowed to go beyond specified dimensions - * (see {@link violin#overflowQ}) + * Gets / sets whether or not legend is allowed to go beyond specified dimensions + * (see {@link legend#spaceX}) * @param {boolean} [_=none] - * @returns {violin | boolean} - * @memberof violin + * @returns {legend | boolean} + * @memberof legend * @property * by default overflowQ = false */ - violin.overflowQ = function(_) { return arguments.length ? (overflowQ = _, violin) : overflowQ; }; - /** - * Gets / sets whether or not to plot points with the violins - * (see {@link violin#pointsQ}) - * @param {boolean} [_=none] - * @returns {violin | boolean} - * @memberof violin - * @property - * by default pointsQ = false - */ - violin.pointsQ = function(_) { return arguments.length ? (pointsQ = _, violin) : pointsQ; }; - - + legend.overflowQ = function(_) { return arguments.length ? (overflowQ = _, legend) : overflowQ; }; /** - * Gets / sets the grouping of the boxes - * (see {@link violin#grouping}) + * Gets / sets the grouping of the bars + * (see {@link legend#grouping}) * @param {Array[]} [_=none] - * @returns {violin | Array[]} - * @memberof violin + * @returns {legend | Array[]} + * @memberof legend * @property * by default grouping = undefined */ - violin.grouping = function(_) { return arguments.length ? (grouping = _, violin) : grouping; }; + legend.grouping = function(_) { return arguments.length ? (grouping = _, legend) : grouping; }; /** * Gets / sets the valueExtractor - * (see {@link violin#valueExtractor}) + * (see {@link legend#valueExtractor}) * @param {function} [_=none] - * @returns {violin | function} - * @memberof violin + * @returns {legend | function} + * @memberof legend * @property + * by default valueExtractor = function(key, index) { return data[key] }, */ - violin.valueExtractor = function(_) { return arguments.length ? (valueExtractor = _, violin) : valueExtractor; }; + legend.valueExtractor = function(_) { return arguments.length ? (valueExtractor = _, legend) : valueExtractor; }; /** * Gets / sets the sortingFunction - * (see {@link violin#sortingFunction}) + * (see {@link bar#sortingFunction}) * @param {function} [_=none] - * @returns {violin | function} - * @memberof violin - * @property - */ - violin.sortingFunction = function(_) { return arguments.length ? (sortingFunction = _, violin) : sortingFunction; }; - - /** - * Gets / sets the scale for which the violin values should be transformed by - * (see {@link violin#scale}) - * @param {d3.scale} [_=none] - * @returns {violin | d3.scale} - * @memberof violin - * @property - * by default scale = d3.scaleLinear() - */ - violin.scale = function(_) { return arguments.length ? (scale = _, violin) : scale; }; - /** - * Gets / sets the padding for the domain of the scale - * (see {@link violin#domainPadding}) - * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin + * @returns {bar | function} + * @memberof bar * @property - * by default domainPadding = 0.5 + * by default sortingFunction = function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}, */ - violin.domainPadding = function(_) { return arguments.length ? (domainPadding = _, violin) : domainPadding; }; - - + legend.sortingFunction = function(_) { return arguments.length ? (sortingFunction = _, legend) : sortingFunction; }; /** * Gets / sets objectSpacer - * (see {@link violin#objectSpacer}) + * (see {@link legend#objectSpacer}) * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin + * @returns {legend | number} + * @memberof legend * @property * by default objectSpacer = 0.05 */ - violin.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, violin) : objectSpacer; }; + legend.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, legend) : objectSpacer; }; /** * Gets / sets the minObjectSize - * (see {@link violin#minObjectSize}) + * (see {@link legend#minObjectSize}) * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin + * @returns {legend | number} + * @memberof legend * @property - * by default minObjectSize = 15 + * by default minObjectSize = 50 */ - violin.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, violin) : minObjectSize; }; + legend.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, legend) : minObjectSize; }; /** * Gets / sets the maxObjectSize - * (see {@link violin#maxObjectSize}) + * (see {@link legend#maxObjectSize}) * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin + * @returns {legend | number} + * @memberof legend * @property - * by default maxObjectSize = 50 + * by default maxObjectSize = 100 */ - violin.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, violin) : maxObjectSize; }; + legend.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, legend) : maxObjectSize; }; /** - * Gets / sets the objectStrokeWidth - * (see {@link violin#objectStrokeWidth}) + * Gets / sets the barStrokeWidth + * (see {@link legend#barStrokeWidth}) * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin + * @returns {legend | number} + * @memberof legend * @property - * by default objectStrokeWidth = 2 + * by default barStrokeWidth = 2 */ - violin.objectStrokeWidth = function(_) { return arguments.length ? (objectStrokeWidth = _, violin) : objectStrokeWidth; }; - - + legend.bubbleStrokeWidth = function(_) { return arguments.length ? (bubbleStrokeWidth = _, legend) : bubbleStrokeWidth; }; /** * Gets / sets the colorFunction - * (see {@link violin#colorFunction}) - * @param {colorFunction} [_=none] - * @returns {violin | colorFunction} - * @memberof violin + * (see {@link legend#colorFunction}) + * @param {number} [_=none] + * @returns {legend | number} + * @memberof legend * @property * by default colorFunction = colorFunction() */ - violin.colorFunction = function(_) { return arguments.length ? (colorFunction$$1 = _, violin) : colorFunction$$1; }; + legend.colorFunction = function(_) { return arguments.length ? (colorFunction$$1 = _, legend) : colorFunction$$1; }; + /** - * Gets / sets the colorFunction - * (see {@link violin#colorFunction}) - * @param {colorFunction} [_=none] - * @returns {violin | colorFunction} - * @memberof violin + * Gets / sets the backgroundFill + * (see {@link legend#backgroundFill}) + * @param {string} [_=none] + * @returns {legend | string} + * @memberof legend * @property - * by default colorFunction = colorFunction() + * by default backgroundFill = 'transparent' */ - violin.pointColorFunc = function(_) { return arguments.length ? (pointColorFunc = _, violin) : pointColorFunc; }; - - + legend.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, legend) : backgroundFill; }; /** - * Gets / sets the pointRadius - * (see {@link violin#pointRadius}) - * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin + * Gets / sets the namespace + * (see {@link legend#namespace}) + * @param {string} [_=none] + * @returns {legend | string} + * @memberof legend * @property - * by default pointRadius = 2 + * by default namespace = 'd3sm-legend' */ - violin.pointRadius = function(_) { return arguments.length ? (pointRadius = _, violin) : pointRadius; }; - /** - * Gets / sets the pointStrokeWidth - * (see {@link violin#pointStrokeWidth}) - * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin - * @property - * by default pointStrokeWidth = 2 - */ - violin.pointStrokeWidth = function(_) { return arguments.length ? (pointStrokeWidth = _, violin) : pointStrokeWidth; }; - - - /** - * Gets / sets the backgroundFill - * (see {@link violin#backgroundFill}) - * @param {string} [_=none] - * @returns {violin | string} - * @memberof violin - * @property - * by default backgroundFill = 'transparent' - */ - violin.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, violin) : backgroundFill; }; - /** - * Gets / sets the namespace - * (see {@link violin#namespace}) - * @param {string} [_=none] - * @returns {violin | string} - * @memberof violin - * @property - * by default namespace = 'd3sm-violin' - */ - violin.namespace = function(_) { return arguments.length ? (namespace = _, violin) : namespace; }; - /** - * Gets / sets the objectClass - * (see {@link violin#objectClass}) - * @param {string} [_=none] - * @returns {violin | string} - * @memberof violin - * @property - * by default objectClass = 'tick-group' - */ - violin.objectClass = function(_) { return arguments.length ? (objectClass = _, violin) : objectClass; }; - - - /** - * Gets / sets the transitionDuration - * (see {@link violin#transitionDuration}) - * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin - * @property - * by default transitionDuration = 1000 - */ - violin.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, violin) : transitionDuration; }; - /** - * Gets / sets the easeFunc - * (see {@link violin#easeFunc}) - * @param {d3.ease} [_=none] - * @returns {violin | d3.ease} - * @memberof violin - * @property - * by default easeFunc = d3.easeExp - */ - violin.easeFunc = function(_) { return arguments.length ? (easeFunc = _, violin) : easeFunc; }; - - - /** - * Gets / sets the quartileKey - * (see {@link violin#quartileKey}) - * @param {string} [_=none] - * @returns {violin | string} - * @memberof violin - * @property - * by default quartileKey = "quartiles" - */ - violin.quartileKey = function(_) { return arguments.length ? (quartileKey = _, violin) : quartileKey; }; - /** - * Gets / sets the quartileKeys - * (see {@link violin#quartileKeys}) - * @param {string[]} [_=none] - * @returns {violin | string[]} - * @memberof violin - * @property - * by default quartileKeys = ["Q0","Q1","Q2","Q3","Q4"] - */ - violin.quartileKeys = function(_) { return arguments.length ? (quartileKeys = _, violin) : quartileKeys; }; - - - /** - * Gets / sets the violinKeys - * (see {@link violin#violinKeys}) - * @param {string[]} [_=none] - * @returns {violin | string[]} - * @memberof violin - * @property - * by default violinKeys = undefined - */ - violin.violinKeys = function(_) { return arguments.length ? (violinKeys = _, violin) : violinKeys; }; - /** - * Gets / sets the violinValues - * (see {@link violin#violinValues}) - * @param {Object[]} [_=none] - * @returns {violin | Object[]} - * @memberof violin - * @property - * by default violinValues = undefined - */ - violin.violinValues = function(_) { return arguments.length ? (violinValues = _, violin) : violinValues; }; - - /** - * Gets / sets the objectSize - * (see {@link violin#objectSize}) - * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin - * @property - * by default objectSize = undefined - */ - violin.objectSize = function(_) { return arguments.length ? (objectSize = _, violin) : objectSize; }; - /** - * Gets / sets the spacerSize - * (see {@link violin#spacerSize}) - * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin - * @property - * by default spacerSize = undefined - */ - violin.spacerSize = function(_) { return arguments.length ? (spacerSize = _, violin) : spacerSize; }; - /** - * Gets / sets the tooltip - * (see {@link violin#tooltip}) - * @param {tooltip} [_=none] - * @returns {violin | tooltip} - * @memberof violin - * @property - * by default tooltip = tooltip() - */ - violin.tooltip = function(_) { return arguments.length ? (tooltip$$1 = _, violin) : tooltip$$1; }; - /** - * Gets / sets the pointsTooltip - * (see {@link violin#pointsTooltip}) - * @param {tooltip} [_=none] - * @returns {violin | tooltip} - * @memberof violin - * @property - * by default pointsTooltip = tooltip() - */ - violin.pointsTooltip = function(_) { return arguments.length ? (pointsTooltip = _, violin) : pointsTooltip; }; - - - - - - function violin () { - // for convenience in handling orientation specific values - var horizontalQ = (orient == 'horizontal') ? true : false; - - // background cliping rectangle - var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; - var container = setupContainer( selection, namespace, bgcpRect, backgroundFill ); - - // if grouping is undefined sort violinKeys by sortingFunction - var ordered = (grouping == undefined) ? d3.keys(data).sort(sortingFunction) : grouping; - - // console.log(ordered) - - violinKeys = flatten(ordered); - - - var calcValues = neededViolinValues() - .horizontalQ(horizontalQ) - .quartileKeys(quartileKeys) - .violinPointsExtractor(violinPointsExtractor) - .violinPointValueExtractor(violinPointValueExtractor); - - - - // augment valus - violinKeys.map(function(vk, i){ calcValues(vk, data); }); - - var numberOfObjects = violinKeys.length; - - var min = [].concat(...violinKeys.map(function(k, i){return data[k].quartiles[quartileKeys[0]]})); - var max = [].concat(...violinKeys.map(function(k, i){return data[k].quartiles[quartileKeys[quartileKeys.length - 1]]})); - var extent = [Math.min(...min) - domainPadding, Math.max(...max) + domainPadding]; - // console.log(extent, violinValues, ordered) - - // set the scale - scale.domain(extent).range(horizontalQ ? [0,spaceY] : [0, spaceX]); - var space = horizontalQ ? spaceX : spaceY; - // calculate object size - objectSize = calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ); - // calculate spacer size if needed - spacerSize = calculateWidthOfSpacer(ordered, space, objectSize, numberOfObjects, objectSpacer, overflowQ); - // make the nested groups - var spacerFunction = groupingSpacer() - .horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects) - .objectClass(objectClass).objectSize(objectSize).spacerSize(spacerSize) - .transitionDuration(transitionDuration).easeFunc(easeFunc) - .namespace(namespace); - - // move stuff - spacerFunction(container, ordered, 0); - // console.log(violinKeys, ordered, container.selectAll('g:not(.to-remove).'+objectClass).nodes()) - - // for color function - var parentIndexArray = []; - container.selectAll('g:not(.to-remove).'+objectClass) - .each(function(d, i){if (hasQ(violinKeys, d)){ parentIndexArray.push(Number(d3.select(this).attr('parent-index')));}}); - - // update color function - colorFunction$$1 = colorFunction$$1.colorBy() == 'index' - ? colorFunction$$1.dataExtent([0, Math.max(...parentIndexArray)]) - : colorFunction$$1.dataExtent(extent); - - /* violiin specific needs */ - - - var frequencyMax = Math.max(...[].concat(...violinKeys.map(function(k, i){return d3.max(data[k].frequencies)}))); - var vScale = d3.scaleLinear().domain([0, frequencyMax]).range([0, objectSize / 2]); - - var lArea = d3.line() - .x(function(d, i){ return horizontalQ ? -vScale(d.x) : scale(d.x)}) - .y(function(d, i){ return horizontalQ ? scale(extent[1]) - scale(d.y) : -vScale(d.y)}) - .curve(d3.curveBasis); - var rArea = d3.line() - .x(function(d, i){ return horizontalQ ? vScale(d.x) : scale(d.x)}) - .y(function(d, i){ return horizontalQ ? scale(extent[1]) - scale(d.y) : vScale(d.y)}) - .curve(d3.curveBasis); - - - - - - - container.selectAll('g:not(.to-remove).'+objectClass).each(function(key, i){ - var t = d3.select(this), - currentData = data[key]; - // needed because bug in selecting .to-remove - if (!hasQ(violinKeys, key)) {return} - var - i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'), - fillColor = colorFunction$$1(key, currentData, i, 'fill'), // prevent duplicate computation - strokeColor = colorFunction$$1(key, currentData, i, 'stroke'), - area = safeSelect(t, 'g', 'area'), - la = safeSelect(area, 'path', 'left'), - ra = safeSelect(area, 'path', 'right'), - quarts = safeSelect(t, 'g', 'quarts'), - lq3 = safeSelect(quarts, 'line', 'q3'), - lq1 = safeSelect(quarts, 'line', 'q1'), - q3 = currentData.quartiles[quartileKeys[3]], - q2 = currentData.quartiles[quartileKeys[2]], - q1 = currentData.quartiles[quartileKeys[1]]; - - t.attr('transform', horizontalQ ? 'translate('+objectSize / 2+',0)' : 'translate(0,'+objectSize / 2+')' ); - // draw curve - la.transition().duration(transitionDuration).attr('d', function(dd, ii){ return lArea(currentData.contour)}) - .attr('fill', fillColor) - .attr('stroke', strokeColor) - .attr('stroke-width', objectStrokeWidth); - - ra.transition().duration(transitionDuration).attr('d', function(dd, ii){ return rArea(currentData.contour)}) - .attr('fill', fillColor) - .attr('stroke', strokeColor) - .attr('stroke-width', objectStrokeWidth); - - area.node().addEventListener('mouseover', function(dd, ii){ - container.selectAll('g.'+objectClass).style('opacity', 0.2); - t.style('opacity', 1); - la.attr('stroke-width',objectStrokeWidth*2); - ra.attr('stroke-width',objectStrokeWidth*2); - }); - area.node().addEventListener('mouseout', function(dd, ii){ - container.selectAll('g.'+objectClass).style('opacity', 1); - la.attr('stroke-width',objectStrokeWidth); - ra.attr('stroke-width',objectStrokeWidth); - }); - - if (pointsQ) { - var ptsContainer = safeSelect(t, 'g', 'points'); - var pts = ptsContainer.selectAll('.point').data(currentData.pointKeys); - pts.on('mouseover', null); - - - var ptsExit = pts.exit().transition().ease(easeFunc).duration(transitionDuration) - .attr('r', 0) - .attr('cy', horizontalQ ? scale(extent[1]) - scale(q2) : vScale(0)) - .attr('cx', horizontalQ ? vScale(0) : scale(q2)).remove(); - - var ptsEnter = pts.enter().append('circle').attr('class', 'point').attr('r', 0) - .attr('cx', horizontalQ ? 0 : scale(q2)) - .attr('cy', horizontalQ ? scale(q2) : 0); - - pts = pts.merge(ptsEnter); - - // console.log(pointsTooltip.header()) - - var pTTips = tooltip().selection(pts).data(violinPointsExtractor(key, currentData)) - .header(pointsTooltip.header()) - .keys(pointsTooltip.keys()) - .values(pointsTooltip.values()); - - pTTips(); - - var pMin = d3.min(currentData.pointValues), pMax = d3.max(currentData.pointValues); - - - - pts.transition().duration(transitionDuration).ease(easeFunc).attr('r', pointRadius) - .attr('cy', function(pointKey, ii){ - var dd = currentData.pointValues[ii]; - if (horizontalQ) { return scale(extent[1]) - scale(dd) } - var j = whichBin(currentData.binned, dd); - var r = Math.random(); - var n = vScale(r * currentData.frequencies[j] * 0.5); - var k = Math.random() > 0.5 ? n : -n; - return k - }) - .attr('cx', function(pointKey, ii){ - var dd = currentData.pointValues[ii]; - if (horizontalQ) { - var j = whichBin(currentData.binned, dd); - var r = Math.random(); - var n = vScale(r * currentData.frequencies[j] * 0.5); - var k = Math.random() > 0.5 ? n : -n; - return k - } - return scale(dd) - }) - .attr('stroke', function(dd, ii) { var dd = currentData.pointValues[ii]; return pointColorFunc(dd, 'stroke', strokeColor, pMin, pMax) }) - .attr('fill' , function(dd, ii) { var dd = currentData.pointValues[ii]; return pointColorFunc(dd, 'fill' , strokeColor, pMin, pMax) }) - .attr('stroke-width', pointStrokeWidth); - - ptsContainer.selectAll('circle.point').on('mouseover', function(dd, ii){ - container.selectAll('g.'+objectClass).style('opacity', 0.2); - t.style('opacity', 1); - la.attr('stroke-width',objectStrokeWidth*2); - ra.attr('stroke-width',objectStrokeWidth*2); - - container.selectAll('.point').style('opacity', 0.2); - d3.select(this).style('opacity', 1).attr('r', pointRadius * 2).attr('stroke-width',pointStrokeWidth*2); - }); - ptsContainer.selectAll('circle.point').on('mouseout', function(dd, ii){ - var e = document.createEvent('SVGEvents'); - e.initEvent('mouseout',true,true); - area.node().dispatchEvent(e); - - container.selectAll('.point').style('opacity', 1); - d3.select(this).attr('stroke-width', pointStrokeWidth).attr('r', pointRadius); - }); - } - else { - cV.selectAll('.point') - .transition().duration(transitionDuration).ease(easeFunc) - .attr('r', 0) - .attr('cy', horizontalQ ? scale(extent[1]) - scale(q2) : vScale(0)) - .attr('cx', horizontalQ ? vScale(0) : scale(q2)) - .remove(); - } - - - }); - - - tooltip$$1.selection(container.selectAll('g:not(.to-remove).'+objectClass + ' .area')); - if (tooltip$$1.data() == undefined) {tooltip$$1.data(data);} - tooltip$$1(); - if (tooltip$$1.values() == undefined) { - tooltip$$1.values([ - function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] }, - function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] }, - function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] }, - function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] }, - function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] } - ]); - - } - - } - - return violin -} - - - - - -/** -* Produces the function which manipulates the violin data to have values needed -* for rendering the violins as svg. -* @returns {function} calculateViolinValues -* @namespace neededViolinValues -*/ -function neededViolinValues() { - var - /** - * Whether or not the orientation of the violins are horizontal - * (see {@link violin#orient}) - * @param {Object} [horizontalQ=true] - * @memberof neededViolinValues# - * @property - */ - horizontalQ = true, - /** - * Keys to be put into the quartiles if they need to be calculated. - * (see {@link violin#quartileKeys}) - * @param {Object} [quartileKeys=['Q0', 'Q1', 'Q2', 'Q3', 'Q4']] - * @memberof neededViolinValues# - * @property - */ - quartileKeys = ['Q0', 'Q1', 'Q2', 'Q3', 'Q4'], - /** - * Function which given the key of the violin and that key's associated value - * returns the object consiting of the points of the violin - * (see {@link violin#violinPointsExtractor}) - * @param {Object} [violinPointsExtractor=function(violinKey, violinData) { return violinData.points }] - * @memberof neededViolinValues# - * @property - */ - violinPointsExtractor, - /** - * Function which given the key of the current point and the object of points for the - * violin, returns the numerical value of the point - * (see {@link violin#violinPointValueExtractor}) - * @param {Object} [violinPointValueExtractor=function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value }] - * @memberof neededViolinValues# - * @property - */ - violinPointValueExtractor; - - - /** - * Gets / sets the horizontalQ - * (see {@link violin#orient}) - * @param {boolean} [_=none] - * @returns {calculateViolinValues | boolean} - * @memberof calculateViolinValues - * @property - * - * by default horizontalQ = true - */ - calculateViolinValues.horizontalQ = function(_) { return arguments.length ? (horizontalQ=_, calculateViolinValues) : horizontalQ }; - /** - * Gets / sets the quartileKeys - * (see {@link violin#quartileKeys}) - * @param {string[]} [_=none] - * @returns {calculateViolinValues | string[]} - * @memberof calculateViolinValues - * @property - * - * by default quartileKeys = ["Q0","Q1","Q2","Q3","Q4"] - */ - calculateViolinValues.quartileKeys = function(_) { return arguments.length ? (quartileKeys=_, calculateViolinValues) : quartileKeys }; - /** - * Gets / sets the violinPointsExtractor - * (see {@link violin#violinPointsExtractor}) - * @param {function} [_=none] - * @returns {calculateViolinValues | function} - * @memberof calculateViolinValues - * @property - * - * by default violinPointsExtractor = function(violinKey, violinData) { return violinData.points } - */ - calculateViolinValues.violinPointsExtractor = function(_) { return arguments.length ? (violinPointsExtractor=_, calculateViolinValues) : violinPointsExtractor }; - /** - * Gets / sets the violinPointValueExtractor - * (see {@link violin#violinPointValueExtractor}) - * @param {function} [_=none] - * @returns {calculateViolinValues | function} - * @memberof calculateViolinValues - * @property - * - * by default violinPointValueExtractor = function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value } - */ - calculateViolinValues.violinPointValueExtractor = function(_) { return arguments.length ? (violinPointValueExtractor=_, calculateViolinValues) : violinPointValueExtractor }; - - - /** - * The function produced by neededViolinValues. - * - * Adds the data need to render the violin as an svg - * @param {string} violinKey the key of the current violin - * @param {object} data the object consisting of violinKey - values (violin data) pairs - * @returns {none} this function manipulates the passed data object adding the following keys - * - * data[violinKey].binned // the binned values of the points - * - * data[violinKey].frequencies // the list consisting of the length of each bin - * - * data[violinKey].contour // the points depicting the contour of 1/2 of the violin - * - * data[violinKey].quartiles // the quartiles of the points' values - * - * data[violinKey].pointKeys // the keys associated with the points - * - * data[violinKey].pointValues // the numerical values of the points - * - * @memberof neededViolinValues# - * @property - */ - function calculateViolinValues(violinKey, data) { - // data for the current violin - var violinData = data[violinKey]; - // the object of points - var violinPoints = violinPointsExtractor(violinKey, violinData); - // - var violinPointsKeys = d3.keys(violinPoints); - // the numerical values of those points - var violinPointsValues = violinPointsKeys.map(function(pk, i){return violinPointValueExtractor(pk, violinPoints)}); - - // quartiles of those points - var pointQuartiles = quartiles(violinPointsValues, quartileKeys); - - // binned points - var binned = d3.histogram()(violinPointsValues); - // length of bins - var frequencies = binned.map(bin=>bin.length); - // min and max countour points for nice drawings - var minContourPoint = horizontalQ ? {x: 0, y: d3.min(violinPointsValues)} : {x: d3.min(violinPointsValues), y: 0}; - var maxContourPoint = horizontalQ ? {x: 0, y: d3.max(violinPointsValues)} : {x: d3.max(violinPointsValues), y: 0}; - var violinContourPoints = binned.map(function(bin, i) { - return horizontalQ - ? {y: (bin.length) ? d3.median(bin): d3.median([bin.x0, bin.x1]), x: frequencies[i]} - : {x: (bin.length) ? d3.median(bin): d3.median([bin.x0, bin.x1]), y: frequencies[i]} - }); - // points along which to draw the violin shpe - violinContourPoints = [minContourPoint].concat(violinContourPoints).concat([maxContourPoint]); - - // set data - violinData.binned = binned; - violinData.frequencies = frequencies; - violinData.contour = violinContourPoints; - violinData.quartiles = pointQuartiles; - violinData.pointKeys = violinPointsKeys; - violinData.pointValues = violinPointsValues; - } - - return calculateViolinValues -} - -function numericLegend( selection ) { - - var - min=0, - max=1, - spaceX, - spaceY, - colorFunction$$1 = colorFunction(), - namespace='d3sm-linear-vertical-gradient', - fontSize = 12, - backgroundFill = 'transparent', - textColor = 'black', - roundTo = 2; - - - legend.min = function(_) { return arguments.length ? (min=_, legend) : min }; - legend.max = function(_) { return arguments.length ? (max=_, legend) : max }; - legend.spaceX = function(_) { return arguments.length ? (spaceX=_, legend) : spaceX }; - legend.spaceY = function(_) { return arguments.length ? (spaceY=_, legend) : spaceY }; - legend.namespace = function(_) { return arguments.length ? (namespace=_, legend) : namespace }; - legend.fontSize = function(_) { return arguments.length ? (fontSize=_, legend) : fontSize }; - legend.backgroundFill = function(_) { return arguments.length ? (backgroundFill=_, legend) : backgroundFill }; - legend.colorFunction = function(_) { return arguments.length ? (colorFunction$$1=_, legend) : colorFunction$$1 }; - legend.textColor = function(_) { return arguments.length ? (textColor=_, legend) : textColor }; - legend.roundTo = function(_) { return arguments.length ? (roundTo=_, legend) : roundTo }; - - function legend() { - // background cliping rectangle - var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; - var container = setupContainer( selection, namespace, bgcpRect, backgroundFill ); - - var defs = safeSelect(selection, 'defs'); - var linearGradient = safeSelect(defs, 'linearGradient') - .attr("x1", "0%") - .attr("y1", "100%") - .attr("x2", "0%") - .attr("y2", "0%") - .attr('id', hypenate(namespace,'numerical-legend-gradient')); - - - colorFunction$$1.dataExtent([min, max]) - .colorBy('value') - .valueExtractor(function(k, v, i){return v}); - - - linearGradient.selectAll('stop') - .data( colorFunction$$1.colors() ) - .enter() - .append('stop') - .attr("offset", function(d, i){ return i / (colorFunction$$1.colors().length - 1) }) - .attr('stop-color', function(d) {return d}); - - - - - var rect = safeSelect(container, 'rect', 'legend') - .attr('transform', 'translate(0,'+fontSize+')') - .style("fill", "url(#"+hypenate(namespace,'numerical-legend-gradient')+")") - .attr('x', 0) - .attr('y', 0) - .attr('width', spaceX) - .attr('height', spaceY - fontSize*2) - .on('mousemove', function(d, i){legendMousemove(d, i, rect, d3.select(this));}) - .on('mouseout', function(d, i){ d3.select("#"+hypenate(namespace,'legend-tooltip')).remove(); }); - - var minText = safeSelect(container, 'text', 'min') - .text(round(min, 2)) - .attr('text-anchor', 'middle') - .attr("font-size", fontSize+'px') - .attr('transform', function(d, i){ - var - x = spaceX / 2, - y = spaceY - fontSize / 4, - t = 'translate('+x+','+y+')'; - return t - }); - - var maxText = safeSelect(container, 'text', 'max') - .text(round(max, 2)) - .attr('text-anchor', 'middle') - .attr("font-size", fontSize+'px') - .attr('transform', function(d, i){ - var - x = spaceX / 2, - y = fontSize, - t = 'translate('+x+','+y+')'; - return t - }); - - - - - } - - function legendMousemove(d, i, rect, t) { - var s = d3.scaleLinear() - .domain([0, rect.attr('height')]) - .range([max, min]); - var m = d3.mouse(rect.node()); - var v = round(s(m[1]),roundTo); - - var strokeColor = colorFunction$$1(undefined, v, undefined, 'stroke'); - var fillColor = colorFunction$$1(undefined, v, undefined, 'fill'); - - var div = safeSelect(d3.select('body'), 'div', hypenate(namespace,'legend-tooltip')) - .attr('id', hypenate(namespace,'legend-tooltip')) - .style('position', 'absolute') - .style('left', (d3.event.pageX+15)+'px') - .style('top', (d3.event.pageY+15)+'px') - .style('background-color', fillColor) - .style('border-color', strokeColor) - - .style('min-width', (fontSize * (String(max).split('.')[0].length+3))+'px') - .style('min-height', (fontSize * (String(max).split('.')[0].length+3))+'px') - .style('border-radius', '50%') - .style('border-radius', '5000px') - - .style('display', 'flex') - .style('justify-content', 'center') - .style('text-align', 'middle') - .style('padding', 2+"px") - - .style('border-style', 'solid') - .style('border-width', 2); - - var text = safeSelect(div, 'div') - .text(v) - .style('color', textColor) - .style('align-self', 'center'); - } - - return legend -} - -function categoricLegend( selection ) { - var - categories, - - /** - * Data to plot. Assumed to be a object, where each key corresponds to a legend - * (see {@link legend#data}) - * @param {Object} [data=undefined] - * @memberof legend# - * @property - */ - data, - /** - * Which direction to render the bars in - * (see {@link legend#orient}) - * @param {number} [orient='horizontal'] - * @memberof legend# - * @property - */ - orient='horizontal', - /** - * Amount of horizontal space (in pixels) avaible to render the legend in - * (see {@link legend#spaceX}) - * @param {number} [spaceX=undefined] - * @memberof legend# - * @property - */ - spaceX, - /** - * Amount of vertical space (in pixels) avaible to render the legend in - * (see {@link legend.spaceY}) - * @param {number} [spaceY=undefined] - * @memberof legend# - * @property - */ - spaceY, - - /** - * Whether or not to allow legend to render elements pass the main spatial dimension - * given the orientation (see {@link legend#orient}), where {@link legend#orient}="horizontal" - * the main dimension is {@link legend#spaceX} and where {@link legend#orient}="vertical" - * the main dimension is {@link legend#spaceY} - * @param {boolean} [overflowQ=false] - * @memberof legend# - * @property - */ - overflowQ = false, - - /** - * An array - putatively of other arrays - depicting how bars should be arranged - * @param {Array[]} [grouping=undefined] - * @memberof legend# - * @property - */ - grouping, - - /** - * How to get the value of the legend - * @param {function} [valueExtractor=function(key, index) { return data[key] }] - * @memberof legend# - * @property - */ - valueExtractor = function(key, index) { return data[key] }, - /** - * How to sort the bars - if {@link bar#grouping} is not provided. - * @param {function} [sortingFunction=function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}] - * @memberof bar# - * @property - */ - sortingFunction = function(keyA, keyB) {return d3.ascending(keyA, keyB)}, - /** - * Default space for the spacer (percentage) of main dimension given the orientation - * (see {@link legend#orient}), where {@link legend#orient}="horizontal" - * the main dimension is {@link legend#spaceX} and where {@link legend#orient}="vertical" - * the main dimension is {@link legend#spaceY} between bars - * @param {number} [objectSpacer=0.05] - * @memberof legend# - * @property - */ - objectSpacer = 0.05, - /** - * The minimum size that an object can be - * @param {number} [minObjectSize=50] - * @memberof legend# - * @property - */ - minObjectSize = 10, - /** - * The maximum size that an object can be - * @param {number} [maxObjectSize=100] - * @memberof legend# - * @property - */ - maxObjectSize = 100, - - /** - * The stroke width of the bars - * @param {number} [barStrokeWidth=2] - * @memberof legend# - * @property - */ - bubbleStrokeWidth = 2, - /** - * Instance of ColorFunction - * @param {function} [colorFunction = colorFunction()] - * @memberof legend# - * @property - */ - colorFunction$$1 = colorFunction(), - - /** - * Color of the background - * @param {string} [backgroundFill="transparent"] - * @memberof legend# - * @property - */ - backgroundFill = 'transparent', - /** - * Namespace for all items made by this instance of legend - * @param {string} [namespace="d3sm-legend"] - * @memberof legend# - * @property - */ - namespace = 'd3sm-legend', - /** - * Class name for legend container ( element) - * @param {string} [objectClass="legend"] - * @memberof legend# - * @property - */ - objectClass = 'legend', - - /** - * Duration of all transitions of this element - * @param {number} [transitionDuration=1000] - * @memberof legend# - * @property - */ - transitionDuration = 1000, - /** - * Easing function for transitions - * @param {d3.ease} [easeFunc=d3.easeExp] - * @memberof legend# - * @property - */ - easeFunc = d3.easeExp; - - - legend.categories = function(_) { return arguments.length ? (categories=_, legend) : categories }; - - /** - * Gets or sets the selection in which items are manipulated - * @param {d3.selection} [_=none] - * @returns {legend | d3.selection} - * @memberof legend - * @property - * by default selection = selection - */ - legend.selection = function(_) { return arguments.length ? (selection = _, legend) : selection; }; - /** - * Gets or sets the data - * (see {@link legend#data}) - * @param {number} [_=none] - * @returns {legend | object} - * @memberof legend - * @property - */ - legend.data = function(_) { return arguments.length ? (data = _, legend) : data; }; - /** - * Gets or sets the orient of the bars - * (see {@link legend#orient}) - * @param {number} [_=none] - * @returns {legend | object} - * @memberof legend - * @property - */ - legend.orient = function(_) { return arguments.length ? (orient = _, legend) : orient; }; - /** - * Gets or sets the amount of horizontal space in which items are manipulated - * (see {@link legend#spaceX}) - * @param {number} [_=none] should be a number > 0 - * @returns {legend | number} - * @memberof legend - * @property - * by default spaceX = undefined - */ - legend.spaceX = function(_) { return arguments.length ? (spaceX = _, legend) : spaceX; }; - /** - * Gets or sets the amount of vertical space in which items are manipulated - * (see {@link legend#spaceY}) - * @param {number} [_=none] should be a number > 0 - * @returns {legend | number} - * @memberof legend - * @property - * by default spaceY = undefined - */ - legend.spaceY = function(_) { return arguments.length ? (spaceY = _, legend) : spaceY; }; - - /** - * Gets / sets whether or not legend is allowed to go beyond specified dimensions - * (see {@link legend#spaceX}) - * @param {boolean} [_=none] - * @returns {legend | boolean} - * @memberof legend - * @property - * by default overflowQ = false - */ - legend.overflowQ = function(_) { return arguments.length ? (overflowQ = _, legend) : overflowQ; }; - /** - * Gets / sets the grouping of the bars - * (see {@link legend#grouping}) - * @param {Array[]} [_=none] - * @returns {legend | Array[]} - * @memberof legend - * @property - * by default grouping = undefined - */ - legend.grouping = function(_) { return arguments.length ? (grouping = _, legend) : grouping; }; - /** - * Gets / sets the valueExtractor - * (see {@link legend#valueExtractor}) - * @param {function} [_=none] - * @returns {legend | function} - * @memberof legend - * @property - * by default valueExtractor = function(key, index) { return data[key] }, - */ - legend.valueExtractor = function(_) { return arguments.length ? (valueExtractor = _, legend) : valueExtractor; }; - /** - * Gets / sets the sortingFunction - * (see {@link bar#sortingFunction}) - * @param {function} [_=none] - * @returns {bar | function} - * @memberof bar - * @property - * by default sortingFunction = function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}, - */ - legend.sortingFunction = function(_) { return arguments.length ? (sortingFunction = _, legend) : sortingFunction; }; - /** - * Gets / sets objectSpacer - * (see {@link legend#objectSpacer}) - * @param {number} [_=none] - * @returns {legend | number} - * @memberof legend - * @property - * by default objectSpacer = 0.05 - */ - legend.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, legend) : objectSpacer; }; - /** - * Gets / sets the minObjectSize - * (see {@link legend#minObjectSize}) - * @param {number} [_=none] - * @returns {legend | number} - * @memberof legend - * @property - * by default minObjectSize = 50 - */ - legend.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, legend) : minObjectSize; }; - /** - * Gets / sets the maxObjectSize - * (see {@link legend#maxObjectSize}) - * @param {number} [_=none] - * @returns {legend | number} - * @memberof legend - * @property - * by default maxObjectSize = 100 - */ - legend.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, legend) : maxObjectSize; }; - - /** - * Gets / sets the barStrokeWidth - * (see {@link legend#barStrokeWidth}) - * @param {number} [_=none] - * @returns {legend | number} - * @memberof legend - * @property - * by default barStrokeWidth = 2 - */ - legend.bubbleStrokeWidth = function(_) { return arguments.length ? (bubbleStrokeWidth = _, legend) : bubbleStrokeWidth; }; - /** - * Gets / sets the colorFunction - * (see {@link legend#colorFunction}) - * @param {number} [_=none] - * @returns {legend | number} - * @memberof legend - * @property - * by default colorFunction = colorFunction() - */ - legend.colorFunction = function(_) { return arguments.length ? (colorFunction$$1 = _, legend) : colorFunction$$1; }; - - /** - * Gets / sets the backgroundFill - * (see {@link legend#backgroundFill}) - * @param {string} [_=none] - * @returns {legend | string} - * @memberof legend - * @property - * by default backgroundFill = 'transparent' - */ - legend.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, legend) : backgroundFill; }; - /** - * Gets / sets the namespace - * (see {@link legend#namespace}) - * @param {string} [_=none] - * @returns {legend | string} - * @memberof legend - * @property - * by default namespace = 'd3sm-legend' - */ - legend.namespace = function(_) { return arguments.length ? (namespace = _, legend) : namespace; }; - /** - * Gets / sets the objectClass - * (see {@link legend#objectClass}) - * @param {string} [_=none] - * @returns {legend | string} - * @memberof legend - * @property - * by default objectClass = 'tick-group' - */ - legend.objectClass = function(_) { return arguments.length ? (objectClass = _, legend) : objectClass; }; - /** - * Gets / sets the transitionDuration - * (see {@link legend#transitionDuration}) - * @param {number} [_=none] - * @returns {legend | number} - * @memberof legend - * @property - * by default transitionDuration = 1000 - */ - legend.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, legend) : transitionDuration; }; - /** - * Gets / sets the easeFunc - * (see {@link legend#easeFunc}) - * @param {d3.ease} [_=none] - * @returns {legend | d3.ease} - * @memberof legend - * @property - * by default easeFunc = d3.easeExp - */ - legend.easeFunc = function(_) { return arguments.length ? (easeFunc = _, legend) : easeFunc; }; - - - function legend() { - var horizontalQ = (orient == 'horizontal') ? true : false; - var verticalQ = !horizontalQ; - // background cliping rectangle - var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; - var container = setupContainer( selection, namespace, bgcpRect, backgroundFill ); - - - colorFunction$$1.dataExtent([0, categories.length - 1]) - .colorBy('categories') - .categoryExtractor(function(k, v, i){return v}); - - var r = Math.min(spaceX, spaceY) / 2; - var numberOfObjects = categories.length; - - // if grouping is undefined sort barKeys by sortingFunction - var ordered = (grouping == undefined) ? categories.sort(sortingFunction) : grouping; - // ordered might be nested depending on grouping - var catKeys = flatten(ordered); - - var space = horizontalQ ? spaceX : spaceY; - // calculate object size - var objectSize = calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ); - // calculate spacer size if needed - var spacerSize = calculateWidthOfSpacer(catKeys, space, objectSize, numberOfObjects, objectSpacer, overflowQ); - // make the nested groups - var spacerFunction = groupingSpacer() - .horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects) - .objectClass(objectClass).objectSize(objectSize).spacerSize(spacerSize) - .transitionDuration(transitionDuration).easeFunc(easeFunc) - .namespace(namespace); - - spacerFunction(container, ordered, 0); - var r = Math.min(objectSize, spaceX, spaceY) / 2 - bubbleStrokeWidth; - - container.selectAll('g:not(.to-remove).'+objectClass).each(function(cat, i) { - var t = d3.select(this); - var c = safeSelect(t, 'circle'); - var fillColor = colorFunction$$1(undefined, cat, i, 'fill'), // prevent duplicate computation - strokeColor = colorFunction$$1(undefined, cat, i, 'stroke'); - - var cx = horizontalQ - ? r+bubbleStrokeWidth - : (spaceX - r*2) / 2 + r; - var cy = verticalQ - ? r+bubbleStrokeWidth - : (spaceX - r*2) / 2 + r; - - c.attr("r", r) - .attr('cx', cx) - .attr('cy', cy) - .attr('fill', fillColor) - .attr('stroke', strokeColor) - .attr('stroke-width', bubbleStrokeWidth); - - var text = safeSelect(t, 'text'); - text.text(cat) - .attr('text-anchor', 'middle') - .attr("transform", function(d, i){ - var - x = cx, - y = cy + text.node().getBoundingClientRect().height / 4, - t = 'translate('+x+','+y+')'; - return t - }); - - }); - - - } - - return legend -} - -// import * as d3 from "d3"; -/******************************************************************************* -** ** -** ** -** D3 EXTENSIONS ** -** ** -** ** -*******************************************************************************/ -/** -* Recursively ascends parents of selection until it finds an svg tag -* @function d3.selection.thisSVG -* @augments d3.selection -* @returns {Element} which is the svg tag, not the d3 selection of that tag -*/ -d3.selection.prototype.thisSVG = function() { return getContainingSVG(this.node()); }; - - -/** -* Helper for getting absolute position of the mouse -* @function d3.mouse.absolute -* @augments d3.mouse -* @returns {number[]} [x, y] as they relate to `html` not to local scope. -*/ -d3.mouse.absolute = function() { - var html = d3.select('html').node(); - var [x, y] = this(html); - return [x, y] -}; - - -/** -* Gets position of the selection in relation to the containing svg -* @see{@link getContainingSVG} -* @function d3.selection.absolutePosition -* @augments d3.selection -* @returns {Object} with structure similar to getBoundingClientRect, e.g. -* top, left, bottom, right, height, width -*/ -d3.selection.prototype.absolutePosition = function() { - var element = this.node(); - var elementPosition = element.getBoundingClientRect(); - var containerSVG = getContainingSVG(element); - var svgPosition = containerSVG.getBoundingClientRect(); - - return { - top: elementPosition.top - svgPosition.top, - left: elementPosition.left - svgPosition.left, - bottom: elementPosition.bottom - svgPosition.top, - right: elementPosition.right - svgPosition.left, - height: elementPosition.height, - width: elementPosition.width - }; - -}; - - -d3.selection.prototype.relativePositionTo = function(container) { - var element = this.node(); - var elementPosition = element.getBoundingClientRect(); - var containerSVG = container; - var svgPosition = containerSVG.getBoundingClientRect(); - - return { - top: elementPosition.top - svgPosition.top, - left: elementPosition.left - svgPosition.left, - bottom: elementPosition.bottom - svgPosition.top, - right: elementPosition.right - svgPosition.left, - height: elementPosition.height, - width: elementPosition.width - }; - -}; - -function getTranslation$1(selection){ - var transform = selection.attr('transform'); - var [junk, xy] =transform.split('translate('); - var [x, y] = xy.split(','); - junk = y.split(')'); - return [parseFloat(x), parseFloat(y)] -} - -function lasso( selection ) { - var - svg, // svg that is target of events - objectContainer, // container which houses objects we are selecting (allows for transform to be applied to lasso) - objectClass, // class of object we are selecting - namespace="d3sm-lasso", - chartContainer, - chartOffset, - objectsOffset, - eventCatcher, - - xScale, // optional scale for the lasso currentPoints - yScale, // optional scale for the lasso currentPoints - - activeQ = false, // whether or not lasso is active - - currentPoints=[], // mouse points for current lasso - allPoints=[], // list of lists for all points of lassos - - line = d3.line() - .x(function(d, i){ - var x; - if (xScale != undefined) { x = xScale(d[0]); } - else {x = d[0];} - return x //- chartOffset[0]// - objectsOffset[0] - }) - .y(function(d, i){ - var y; - if (yScale != undefined) { y= yScale(d[1]); } - else {y = d[1];} - return y// - chartOffset[1]// - objectsOffset[1] - }) - .curve(d3.curveLinearClosed), - - instance=0, // an indentifier for which instance this lasso is under the current svg - - tickDistance = 10, - - // styles for lasso path - color = '#17a2b8', - animationRate = '10s', - opacity=0.3, - dashArray = '5, 10', - stroke = 'black', - strokeWidth=2, - - // styles for lassoed objects - lassoedFill = "white", - lassoedStroke = 'black', - lassoedStrokeWidth = 3, - - transitionDuration = 1000, - easeFunc = d3.easeExp; - - var path; - - lasso.svg = function(_) { return arguments.length ? (svg = _, lasso) : svg; }; - lasso.chartContainer = function(_) { return arguments.length ? (chartContainer = _, lasso) : chartContainer; }; - lasso.objectContainer = function(_) { return arguments.length ? (objectContainer = _, lasso) : objectContainer; }; - lasso.objectClass = function(_) { return arguments.length ? (objectClass = _, lasso) : objectClass; }; - lasso.namespace = function(_) { return arguments.length ? (namespace = _, lasso) : namespace; }; - lasso.xScale = function(_) { return arguments.length ? (xScale = _, lasso) : xScale; }; - lasso.yScale = function(_) { return arguments.length ? (yScale = _, lasso) : yScale; }; - lasso.activeQ = function(_) { return arguments.length ? (activeQ = _, lasso) : activeQ; }; - lasso.currentPoints = function(_) { return arguments.length ? (currentPoints = _, lasso) : currentPoints; }; - lasso.allPoints = function(_) { return arguments.length ? (allPoints = _, lasso) : allPoints; }; - lasso.instance = function(_) { return arguments.length ? (instance = _, lasso) : instance; }; - lasso.tickDistance = function(_) { return arguments.length ? (tickDistance = _, lasso) : tickDistance; }; - lasso.color = function(_) { return arguments.length ? (color = _, lasso) : color; }; - lasso.animationRate = function(_) { return arguments.length ? (animationRate = _, lasso) : animationRate; }; - lasso.opacity = function(_) { return arguments.length ? (opacity = _, lasso) : opacity; }; - lasso.dashArray = function(_) { return arguments.length ? (dashArray = _, lasso) : dashArray; }; - lasso.stroke = function(_) { return arguments.length ? (stroke = _, lasso) : stroke; }; - lasso.lassoedFill = function(_) { return arguments.length ? (lassoedFill = _, lasso) : lassoedFill; }; - lasso.lassoedStroke = function(_) { return arguments.length ? (lassoedStroke = _, lasso) : lassoedStroke; }; - lasso.lassoedStrokeWidth = function(_) { return arguments.length ? (lassoedStrokeWidth = _, lasso) : lassoedStrokeWidth; }; - lasso.eventCatcher = function(_) { return arguments.length ? (eventCatcher = _, lasso) : eventCatcher; }; - - lasso.drag = drag; - lasso.draw = draw; - lasso.tick = tick; - lasso.detect = detect; - lasso.toggle = toggle; - lasso.remove = remove; - lasso.render = render; - lasso.keyFrames = keyFrames; - lasso.updateObjects = updateObjects; - lasso.applyPathAttributes = applyPathAttributes; - lasso.applyObjectAttributes = applyObjectAttributes; - - keyFrames(); - - function lasso() { - // add a dash animation if needed - if (activeQ) { transitionDraw(); } - } - - function toggle(state) { - // use optional param to set state, otherwise toggle state - activeQ = (state!=undefined) ? state : !activeQ; - chartOffset = getTranslation$1(chartContainer); - objectsOffset = getTranslation$1(objectContainer); - - if (activeQ) { - svg.node().addEventListener('mousedown', render, true); - } else { - svg.node().removeEventListener('mousedown', render, true); - remove(); - } - - } - - function draw() { - chartOffset = getTranslation$1(chartContainer); - objectsOffset = getTranslation$1(objectContainer); - - var container = safeSelect(objectContainer, 'g', 'lasso-container'); - var paths = container.selectAll('path[instance="'+instance+'"]'); - - // update - paths = paths.data(allPoints); - - // remove excess - var pExit = paths.exit().remove(); - // add needed paths - var pEnter = paths.enter().append('path'); - - // merge - paths = paths.merge(pEnter); - - // apply - applyPathAttributes(paths); - } - - function remove() { - var container = safeSelect(objectContainer, 'g', 'lasso-container'); - var paths = container.selectAll('path[instance="'+instance+'"]').remove(); - container.remove(); - objectContainer.selectAll(objectClass).classed("in-lasso", false); - updateObjects(); - } - - function render( event ) { - // nothing can interefer with drawing the lasso - event.preventDefault(); event.stopPropagation(); - - var container = safeSelect(objectContainer, 'g', 'lasso-container'); - - /* - each time the user presses down, while the state is active, the lasso - the lasso should make a seperate segment. - */ - currentPoints = []; - - svg.node().addEventListener('mousemove', drag); - svg.node().addEventListener('mouseup', function(event) { - svg.node().removeEventListener('mousemove', drag); - allPoints.push(currentPoints); - // BUG: somehow this is pushing currentPoints n times where n is the nth lasso path for the current instance - // NOTE: allPoints = unique(allPoints) is a temporary and inefficient fix - allPoints = unique(allPoints); - }); - - path = container.append('path').data([currentPoints]); - applyPathAttributes(path); - } - - function transitionDraw() { - var container = safeSelect(objectContainer, 'g', 'lasso-container'); - var paths = container.selectAll('path[instance="'+instance+'"]'); - - // update - paths = paths.data(allPoints); - - // remove excess - var pExit = paths.exit().remove(); - // add needed paths - var pEnter = paths.enter().append('path'); - - // merge - paths = paths.merge(pEnter) - .transition().duration(transitionDuration) - .ease(easeFunc); - applyPathAttributes(paths); - - } - - function applyPathAttributes(path) { - path - .attr("class", hypenate(namespace, "lasso-path")) - .style('opacity', opacity) - .attr('fill', color) - .attr("d", line) - .attr('instance', instance) - .style("stroke-dasharray", dashArray) - .attr("stroke", stroke) - .attr("stroke-width", strokeWidth) - .style('animation', 'lassoDash '+animationRate+' linear') - .style("animation-iteration-count", "infinite"); - } - - function drag(event) { - /* - effectively create a mouse down and move event (which normally is inteperated - as 'drag' by the browser) by dynamically adding / removing this event on - mouse down / mouse up. - */ - - if (eventCatcher != undefined) {eventCatcher.dispatch(hypenate(namespace,"drag"));} - // d3.dispatch(hypenate(namespace,"drag")) - - if (event.which != 1) {return} // ensures left mouse button set - d3.event = event; - var pt = d3.mouse(objectContainer.node()); - var pt = d3.mouse(svg.node()); - - if (xScale != undefined) {pt[0] = xScale.invert(pt[0]);} - if (yScale != undefined) {pt[1] = yScale.invert(pt[1]);} - pt[0] = pt[0] - chartOffset[0] - objectsOffset[0]; - pt[1] = pt[1] - chartOffset[1] - objectsOffset[1]; - - /* if we have a point already, test if it passes a minimum distance to prevent overwhelming with too many tick functions */ - if (currentPoints.length) { - var lastPt = currentPoints[currentPoints.length - 1]; - var a = [pt[0], pt[1]], b = [lastPt[0], lastPt[1]]; - - if (xScale) {b[0] = xScale(b[0]); a[0] = xScale(a[0]);} - if (yScale) {b[1] = yScale(b[1]); a[1] = yScale(a[1]);} - - var dist = euclideanDistance(b, a); - if (dist > tickDistance) { tick(pt); } - } - else { tick(pt); } - } + legend.namespace = function(_) { return arguments.length ? (namespace = _, legend) : namespace; }; + /** + * Gets / sets the objectClass + * (see {@link legend#objectClass}) + * @param {string} [_=none] + * @returns {legend | string} + * @memberof legend + * @property + * by default objectClass = 'tick-group' + */ + legend.objectClass = function(_) { return arguments.length ? (objectClass = _, legend) : objectClass; }; + /** + * Gets / sets the transitionDuration + * (see {@link legend#transitionDuration}) + * @param {number} [_=none] + * @returns {legend | number} + * @memberof legend + * @property + * by default transitionDuration = 1000 + */ + legend.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, legend) : transitionDuration; }; + /** + * Gets / sets the easeFunc + * (see {@link legend#easeFunc}) + * @param {d3.ease} [_=none] + * @returns {legend | d3.ease} + * @memberof legend + * @property + * by default easeFunc = d3.easeExp + */ + legend.easeFunc = function(_) { return arguments.length ? (easeFunc = _, legend) : easeFunc; }; - function tick (pt) { - /* - If a point is provided update data and objects. - Otherwise just call on data we already have. + function legend() { + var horizontalQ = (orient == 'horizontal') ? true : false; + var verticalQ = !horizontalQ; + // background cliping rectangle + var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; + var container = utils$1.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); - Why like this?: - 1. currentPoints is current points to allow disjunct lassos, currentPoints is only pushed to - allPoints after mouseup. - 2. to allow render of objects in the lasso class / updating the data list - just by toggling the button - */ - if (pt != undefined) { - currentPoints.push(pt); - path.attr("d", line); - if (currentPoints.length < 3) {return} // need at least 3 points to detect anything. - detect(allPoints.concat([currentPoints])); - } else { - detect(allPoints); - } - } + colorFunction$$1.dataExtent([0, categories.length - 1]) + .colorBy('categories') + .categoryExtractor(function(k, v, i){return v}); + var r = Math.min(spaceX, spaceY) / 2; + var numberOfObjects = categories.length; - function detect(lassos) { - if (lassos == undefined) {lassos = allPoints;} - objectContainer.selectAll(objectClass).each(function(d, i){ - var current = d3.select(this), + // if grouping is undefined sort barKeys by sortingFunction + var ordered = (grouping == undefined) ? categories.sort(sortingFunction) : grouping; + // ordered might be nested depending on grouping + var catKeys = utils$1.arr.flatten(ordered); - box = current.absolutePosition(), - // box = current.relativePositionTo(objectContainer.node()), + var space = horizontalQ ? spaceX : spaceY; + // calculate object size + var objectSize = utils$1.math.calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ); + // calculate spacer size if needed + var spacerSize = utils$1.math.calculateWidthOfSpacer(catKeys, space, objectSize, numberOfObjects, objectSpacer, overflowQ); + // make the nested groups + var spacerFunction = groupingSpacer() + .horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects) + .objectClass(objectClass).objectSize(objectSize).spacerSize(spacerSize) + .transitionDuration(transitionDuration).easeFunc(easeFunc) + .namespace(namespace); - boxPts = [ - [ - box.left - chartOffset[0] - objectsOffset[0], - box.top - chartOffset[1] - objectsOffset[1] - ], - [ - box.right - chartOffset[0] - objectsOffset[0], - box.top - chartOffset[1] - objectsOffset[1] - ], - [ - box.left - chartOffset[0] - objectsOffset[0], - box.bottom - chartOffset[1] - objectsOffset[1] - ], - [ - box.right - chartOffset[0] - objectsOffset[0], - box.bottom - chartOffset[1] - objectsOffset[1] - ] - ]; + spacerFunction(container, ordered, 0); + var r = Math.min(objectSize, spaceX, spaceY) / 2 - bubbleStrokeWidth; - if (xScale != undefined) { - boxPts[0][0] = xScale.invert(boxPts[0][0]); - boxPts[1][0] = xScale.invert(boxPts[1][0]); - boxPts[2][0] = xScale.invert(boxPts[2][0]); - boxPts[3][0] = xScale.invert(boxPts[3][0]); - } - if (yScale != undefined) { - boxPts[0][1] = yScale.invert(boxPts[0][1]); - boxPts[1][1] = yScale.invert(boxPts[1][1]); - boxPts[2][1] = yScale.invert(boxPts[2][1]); - boxPts[3][1] = yScale.invert(boxPts[3][1]); - } + container.selectAll('g:not(.to-remove).'+objectClass).each(function(cat, i) { + var t = d3.select(this); + var c = utils$1.sel.safeSelect(t, 'circle'); + var fillColor = colorFunction$$1(undefined, cat, i, 'fill'), // prevent duplicate computation + strokeColor = colorFunction$$1(undefined, cat, i, 'stroke'); + var cx = horizontalQ + ? r+bubbleStrokeWidth + : (spaceX - r*2) / 2 + r; + var cy = verticalQ + ? r+bubbleStrokeWidth + : (spaceX - r*2) / 2 + r; - /* - flag needed as we have to test multiple lasso segments, and if the point - is not in one segment, it does not mean it is not in any - */ - var inAnyLassoQ = false; - for (var i = 0; i < lassos.length; i++) { - var lassoPoints = lassos[i]; - // .map(function(pt){ - // var x, y = pt - // if (xScale!=undefined) {x = xScale(x)} - // if (yScale!=undefined) {y = yScale(y)} - // return [x, y] - // }) - var boxInLassoQ = boxPts.every(coord => d3.polygonContains(lassoPoints, coord)); + c.attr("r", r) + .attr('cx', cx) + .attr('cy', cy) + .attr('fill', fillColor) + .attr('stroke', strokeColor) + .attr('stroke-width', bubbleStrokeWidth); - if (boxInLassoQ) { inAnyLassoQ = true; } // only update flag in the positive case. - } + var text = utils$1.sel.safeSelect(t, 'text'); + text.text(cat) + .attr('text-anchor', 'middle') + .attr("transform", function(d, i){ + var + x = cx, + y = cy + text.node().getBoundingClientRect().height / 4, + t = 'translate('+x+','+y+')'; + return t + }); - current.classed('in-lasso', inAnyLassoQ); - current.classed('in-lasso-'+instance, inAnyLassoQ); }); - updateObjects(); - return objectContainer.selectAll('.in-lasso-'+instance) - } + } + return legend +} - function updateObjects() { - objectContainer.selectAll(objectClass).each(function(d, i) { - var t = d3.select(this); - applyObjectAttributes(t, t.classed('in-lasso')); - }); - } +function numericLegend( selection ) { - function applyObjectAttributes(obj, setQ) { - var - preLassoFill = obj.attr('_pre_lasso_fill'), - preLassoStroke = obj.attr('_pre_lasso_stroke'), - preLassoStrokeWidth = obj.attr('_pre_lasso_stroke-width'); + var + min=0, + max=1, + spaceX, + spaceY, + colorFunction$$1 = colorFunction(), + namespace='d3sm-linear-vertical-gradient', + fontSize = 12, + backgroundFill = 'transparent', + textColor = 'black', + roundTo = 2; - if (setQ) { - obj.classed("in-lasso", true); - obj.classed('in-lasso-'+instance, true); - if (preLassoFill == undefined) { obj.attr('_pre_lasso_fill', obj.attr('fill')); } - if (preLassoStroke == undefined) { obj.attr('_pre_lasso_stroke', obj.attr('stroke')); } - if (preLassoStrokeWidth == undefined) { obj.attr('_pre_lasso_stroke-width', obj.attr('stroke-width')); } - obj - //BUG: when .raise() - .attr('fill', lassoedFill) - .attr('stroke', lassoedStroke) - .attr('stoke-width', lassoedStrokeWidth); + legend.min = function(_) { return arguments.length ? (min=_, legend) : min }; + legend.max = function(_) { return arguments.length ? (max=_, legend) : max }; + legend.spaceX = function(_) { return arguments.length ? (spaceX=_, legend) : spaceX }; + legend.spaceY = function(_) { return arguments.length ? (spaceY=_, legend) : spaceY }; + legend.namespace = function(_) { return arguments.length ? (namespace=_, legend) : namespace }; + legend.fontSize = function(_) { return arguments.length ? (fontSize=_, legend) : fontSize }; + legend.backgroundFill = function(_) { return arguments.length ? (backgroundFill=_, legend) : backgroundFill }; + legend.colorFunction = function(_) { return arguments.length ? (colorFunction$$1=_, legend) : colorFunction$$1 }; + legend.textColor = function(_) { return arguments.length ? (textColor=_, legend) : textColor }; + legend.roundTo = function(_) { return arguments.length ? (roundTo=_, legend) : roundTo }; - } else { - obj.classed("in-lasso", false); - obj.classed('in-lasso-'+instance, false); - if (preLassoFill != undefined) { obj.attr('fill', preLassoFill); } - if (preLassoStroke != undefined) { obj.attr('stroke', preLassoStroke); } - if (preLassoStrokeWidth != undefined) { obj.attr('stroke-width', preLassoStrokeWidth); } - } - } + function legend() { + // background cliping rectangle + var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; + var container = utils$1.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); - function keyFrames() { - var style = - d3.select("html").select('style.'+hypenate(namespace,"lasso-dash")); - if (style.empty()) { - d3.select("html").append('style') - .classed(hypenate(namespace,"lasso-dash"), true) - .html("@keyframes lassoDash {to { stroke-dashoffset: 1000;}}"); - } + var defs = utils$1.sel.safeSelect(selection, 'defs'); + var linearGradient = utils$1.sel.safeSelect(defs, 'linearGradient') + .attr("x1", "0%") + .attr("y1", "100%") + .attr("x2", "0%") + .attr("y2", "0%") + .attr('id', utils$1.str.hypenate(namespace,'numerical-legend-gradient')); - } - return lasso -} -function lassoWidget( selection ) { - var - namespace = 'd3sm-lasso', - selection = selection, - svg, - chartContainer, - objectContainer, - objectClass, - lassoLine = d3.line() - .x(function(d, i){return d[0]}) - .y(function(d, i){return d[1]}) - .curve(d3.curveLinearClosed), - colorFunction$$1 = colorFunction(), - maxNumberOfGroups, - dataExtractor, - xScale, - yScale; + colorFunction$$1.dataExtent([min, max]) + .colorBy('value') + .valueExtractor(function(k, v, i){return v}); - function onError(msg, style){ - console.log(msg, style); - } - // setup the container - setupDataselectContainer(); - - lassoWidget.objectClass = function(_){return arguments.length ? (objectClass='.'+_.replace('.',''), lassoWidget) : objectClass}; - lassoWidget.svg = function(_){return arguments.length ? (svg=_, lassoWidget) : svg}; - lassoWidget.submit = function(_){return arguments.length ? (submit=_, lassoWidget) : submit}; - lassoWidget.maxNumberOfGroups = function(_){return arguments.length ? (maxNumberOfGroups=_, lassoWidget) : maxNumberOfGroups}; - lassoWidget.onError = function(_){return arguments.length ? (onError=_, lassoWidget) : onError}; - lassoWidget.objectContainer = function(_){return arguments.length ? (objectContainer=_, lassoWidget) : objectContainer}; - lassoWidget.chartContainer = function(_){return arguments.length ? (chartContainer=_, lassoWidget) : chartContainer}; - lassoWidget.dataExtractor = function(_){return arguments.length ? (dataExtractor=_, lassoWidget) : dataExtractor}; - lassoWidget.xScale = function(_){return arguments.length ? (xScale=_, lassoWidget) : xScale}; - lassoWidget.yScale = function(_){return arguments.length ? (yScale=_, lassoWidget) : yScale}; - - function styles() { - var s = d3.select("html").select("style."+namespace+'lasso-widget'); - if (s.empty()) { - d3.select("html").append("style") - .classed(namespace+'lasso-widget', true) - .html( - "."+hypenate(namespace, "data-table") + "{\ - height:100px;\ - overflow:auto;\ - }" - ); - } - } + linearGradient.selectAll('stop') + .data( colorFunction$$1.colors() ) + .enter() + .append('stop') + .attr("offset", function(d, i){ return i / (colorFunction$$1.colors().length - 1) }) + .attr('stop-color', function(d) {return d}); - function lassoWidget() { - styles(); - } - function submit(data) { - console.log(data); - } - function plusTab(tabList) { - var tab = safeSelect(tabList, 'li', hypenate(namespace, 'plus-tab')) - .classed('ml-auto', true) - .classed('nav-item', true) - .on('click', makeNewGroup), + var rect = utils$1.sel.safeSelect(container, 'rect', 'legend') + .attr('transform', 'translate(0,'+fontSize+')') + .style("fill", "url(#"+utils$1.str.hypenate(namespace,'numerical-legend-gradient')+")") + .attr('x', 0) + .attr('y', 0) + .attr('width', spaceX) + .attr('height', spaceY - fontSize*2) + .on('mousemove', function(d, i){legendMousemove(d, i, rect, d3.select(this));}) + .on('mouseout', function(d, i){ d3.select("#"+utils$1.str.hypenate(namespace,'legend-tooltip')).remove(); }); - anchor = safeSelect(tab, 'a', 'nav-link'), - icon = safeSelect(anchor, 'i', 'fa') - .classed('fa-plus fa-2x text-success', true); - } + var minText = utils$1.sel.safeSelect(container, 'text', 'min') + .text(utils$1.math.round(min, 2)) + .attr('text-anchor', 'middle') + .attr("font-size", fontSize+'px') + .attr('transform', function(d, i){ + var + x = spaceX / 2, + y = spaceY - fontSize / 4, + t = 'translate('+x+','+y+')'; + return t + }); - function sendTab(tabList) { - var tab = safeSelect(tabList, 'li', hypenate(namespace, 'send-tab')) - .classed('ml-auto', true) - .classed('nav-item', true) - .on('click', clickSend) - , + var maxText = utils$1.sel.safeSelect(container, 'text', 'max') + .text(utils$1.math.round(max, 2)) + .attr('text-anchor', 'middle') + .attr("font-size", fontSize+'px') + .attr('transform', function(d, i){ + var + x = spaceX / 2, + y = fontSize, + t = 'translate('+x+','+y+')'; + return t + }); - anchor = safeSelect(tab, 'a', 'nav-link'), - icon = safeSelect(anchor, 'i', 'fa') - .classed('fa-paper-plane-o fa-2x text-primary', true); - } - function clickSend() { - var data = gatherDataLists(); - submit(data); - } - function closeTab(tabList) { - var tab = safeSelect(tabList, 'li', hypenate(namespace, 'close-tab')) - .classed('ml-auto', true) - .classed('nav-item', true) - .on('click', function(){ - var m = d3.mouse(d3.select("html").node()); - selection.select('div.card').style("position", 'relative') - .transition().duration(1000) - .ease(d3.easeBack) - .style('left', window.outerWidth +'px') - .remove(); - }) - , - anchor = safeSelect(tab, 'a', 'nav-link'), - icon = safeSelect(anchor, 'i', 'fa') - .classed('fa-window-close-o fa-2x text-danger', true); } + function legendMousemove(d, i, rect, t) { + var s = d3.scaleLinear() + .domain([0, rect.attr('height')]) + .range([max, min]); + var m = d3.mouse(rect.node()); + var v = utils$1.math.round(s(m[1]),roundTo); + var strokeColor = colorFunction$$1(undefined, v, undefined, 'stroke'); + var fillColor = colorFunction$$1(undefined, v, undefined, 'fill'); - function setupDataselectContainer() { - var - card = safeSelect(selection, 'div', 'card'), - cardHeader = safeSelect(card, 'div', 'card-header'), - - tabList = safeSelect(cardHeader, 'ul', 'nav') - .classed('nav-tabs card-header-tabs', true) - .classed(hypenate(namespace, 'tab-list'), true) - .attr('role', 'tablist'), - - cardBody = safeSelect(card, 'div', 'card-body'), - tabContent = safeSelect(cardBody, 'div', 'tab-content') - .classed(hypenate(namespace, 'tab-content'), true), + var div = utils$1.sel.safeSelect(d3.select('body'), 'div', utils$1.str.hypenate(namespace,'legend-tooltip')) + .attr('id', utils$1.str.hypenate(namespace,'legend-tooltip')) + .style('position', 'absolute') + .style('left', (d3.event.pageX+15)+'px') + .style('top', (d3.event.pageY+15)+'px') + .style('background-color', fillColor) + .style('border-color', strokeColor) - // leftTabs = safeSelect(tabList, 'div', 'nav mr-auto left-aligned-tabs') - rightTabs = safeSelect(tabList, 'div', 'right-aligned-tabs').classed('nav ml-auto ', true); + .style('min-width', (fontSize * (String(max).split('.')[0].length+3))+'px') + .style('min-height', (fontSize * (String(max).split('.')[0].length+3))+'px') + .style('border-radius', '50%') + .style('border-radius', '5000px') - plusTab(rightTabs); - sendTab(rightTabs); - closeTab(rightTabs); - var - defaultTab = safeSelect(tabContent, 'div', 'tab-pane') - .classed(hypenate(namespace,'default-tab'), true) - .classed("active", true) - .classed('text-left', true), - - p = safeSelect(defaultTab, 'div') - .html( - "Click to add a new group.
"+ - "Click to submit for re-analysis.
"+ - "Click to close the dataselect widget." - ); - // .text('Click the green plus to add a new group.') - } + .style('display', 'flex') + .style('justify-content', 'center') + .style('text-align', 'middle') + .style('padding', 2+"px") + .style('border-style', 'solid') + .style('border-width', 2); - function makeRemoveButton(paneBtnList) { - var - btn = safeSelect(paneBtnList, 'button', 'remove-btn') - .classed('btn btn-danger', true) - .on('click', removeBtnClickToRemove), - i = safeSelect(btn, 'i', 'fa').classed('fa-trash-o', true), - s = safeSelect(btn, 'span').text('Remove group'); - return btn + var text = utils$1.sel.safeSelect(div, 'div') + .text(v) + .style('color', textColor) + .style('align-self', 'center'); } - function removeBtnClickToRemove(d, i) { - var - tab = selection.select('#'+hypenate(namespace,'tab',d)), - pane = selection.select('#'+hypenate(namespace,'tab','pane',d)); + return legend +} - tab.remove(); - pane.remove(); +let legends = { + categorical: categoricLegend, numeric: numericLegend +}; - showRemainingPanes(); - updateTabAndPaneNumbers(); - } +// import * as d3 from "d3"; +/******************************************************************************* +** ** +** ** +** D3 EXTENSIONS ** +** ** +** ** +*******************************************************************************/ +/** +* Recursively ascends parents of selection until it finds an svg tag +* @function d3.selection.thisSVG +* @augments d3.selection +* @returns {Element} which is the svg tag, not the d3 selection of that tag +*/ +d3.selection.prototype.thisSVG = function() { return utils$1.sel.getContainingSVG(this.node()); }; - function togglePaneById(id) { - var panes = selection.select('.'+hypenate(namespace, 'tab-content')); - panes.selectAll('div.tab-pane') - .each(function(d, i){ - d3.select(this).classed('active show', d3.select(this).attr('id') == id); - }); - } +/** +* Helper for getting absolute position of the mouse +* @function d3.mouse.absolute +* @augments d3.mouse +* @returns {number[]} [x, y] as they relate to `html` not to local scope. +*/ +d3.mouse.absolute = function() { + var html = d3.select('html').node(); + var [x, y] = this(html); + return [x, y] +}; - function insertTab(tabs, n) { - - // var li = tabs.insert("li", ':nth-child('+(n)+')') - var li = tabs.insert("li", '.right-aligned-tabs') - .classed('nav-item', true) - .classed('alert', true) - .classed('alert-secondary', true) - .classed('alert-dismissable', true) - .classed('fade', true) - .classed('show', true) - // .classed('mr-auto', true) - .attr("role", 'alert') - .attr("id", hypenate(namespace,'tab',n)) - .classed(hypenate(namespace,'tab'), true); - - var a = safeSelect(li, 'a') - .attr('data-toggle', 'tab') - .text('Group ' + n) - .attr('href', '#'+hypenate(namespace,'tab','pane',n)) - .on('dblclick', function(d, i){ - var t = d3.select(this); - t.attr('contenteditable', true); - d3.select(t.node().parentNode) - .classed('alert-secondary', false) - .classed('alert-warning', true); - // d3.select("html").on("click", dispatchBlur) - // - // function dispatchBlur(d, i) { - // if (d3.event.target != d3.select(this)) { - // d3.select(this).dispatch("blur") - // } - // } +/** +* Gets position of the selection in relation to the containing svg +* @see{@link getContainingSVG} +* @function d3.selection.absolutePosition +* @augments d3.selection +* @returns {Object} with structure similar to getBoundingClientRect, e.g. +* top, left, bottom, right, height, width +*/ +d3.selection.prototype.absolutePosition = function() { + var element = this.node(); + var elementPosition = element.getBoundingClientRect(); + var containerSVG = getContainingSVG(element); + var svgPosition = containerSVG.getBoundingClientRect(); + return { + top: elementPosition.top - svgPosition.top, + left: elementPosition.left - svgPosition.left, + bottom: elementPosition.bottom - svgPosition.top, + right: elementPosition.right - svgPosition.left, + height: elementPosition.height, + width: elementPosition.width + }; - }) - .on('blur', function(d, i){ +}; - var t = d3.select(this); - t.attr('contenteditable', false); - d3.select(t.node().parentNode) - .classed('alert-secondary', true) - .classed('alert-warning', false); - }) - .on('input', function(d, i){ - var t = d3.select(this); - var txt = t.text(); - d3.select(t.attr('href')).select("p.lead").text(txt); - }); +d3.selection.prototype.relativePositionTo = function(container) { + var element = this.node(); + var elementPosition = element.getBoundingClientRect(); + var containerSVG = container; + var svgPosition = containerSVG.getBoundingClientRect(); + return { + top: elementPosition.top - svgPosition.top, + left: elementPosition.left - svgPosition.left, + bottom: elementPosition.bottom - svgPosition.top, + right: elementPosition.right - svgPosition.left, + height: elementPosition.height, + width: elementPosition.width + }; - return li - } +}; - function makeTabPane(panes, n) { - var pane = panes.append('div') - .datum(n) - .attr('role', 'tabpanel') - .classed('tab-pane', true) - .classed('text-left', true) - .classed(hypenate(namespace,'tab','pane'), true) - .attr('id', hypenate(namespace,'tab','pane',n)); - return pane - } +// function getTranslation(selection){ +// var transform = selection.attr('transform') +// var [junk, xy] =transform.split('translate(') +// var [x, y] = xy.split(',') +// y, junk = y.split(')') +// return [parseFloat(x), parseFloat(y)] +// } - function populatePane(pane, n) { - var - lead = safeSelect(pane, 'p', 'lead').text('Group '+n), +function lasso( selection ) { + var + svg, // svg that is target of events + objectContainer, // container which houses objects we are selecting (allows for transform to be applied to lasso) + objectClass, // class of object we are selecting + namespace="d3sm-lasso", + chartContainer, + chartOffset, + objectsOffset, + eventCatcher, - tableContainer = safeSelect(pane, 'div', 'table-responsive') - .attr("class", hypenate(namespace, "data-table")), + xScale, // optional scale for the lasso currentPoints + yScale, // optional scale for the lasso currentPoints - table = safeSelect(tableContainer, "table", "table") - .classed("table-sm", true).classed("table-hover", true), + activeQ = false, // whether or not lasso is active - caption = safeSelect(table, "caption", "caption").html("List of selected"), + currentPoints=[], // mouse points for current lasso + allPoints=[], // list of lists for all points of lassos - btns = safeSelect(pane, 'div', 'text-right'), + line = d3.line() + .x(function(d, i){ + var x; + if (xScale != undefined) { x = xScale(d[0]); } + else {x = d[0];} + return x //- chartOffset[0]// - objectsOffset[0] + }) + .y(function(d, i){ + var y; + if (yScale != undefined) { y= yScale(d[1]); } + else {y = d[1];} + return y// - chartOffset[1]// - objectsOffset[1] + }) + .curve(d3.curveLinearClosed), - cN = n % colorFunction$$1.colors().length; - if (n % 2 != 0) { cN = (colorFunction$$1.colors().length - 1) - cN; } + instance=0, // an indentifier for which instance this lasso is under the current svg + tickDistance = 10, + // styles for lasso path + color$$1 = '#17a2b8', + animationRate = '10s', + opacity=0.3, + dashArray = '5, 10', + stroke = 'black', + strokeWidth=2, + // styles for lassoed objects + lassoedFill = "white", + lassoedStroke = 'black', + lassoedStrokeWidth = 3, + transitionDuration = 1000, + easeFunc = d3.easeExp; - var lassoBtn = makeLassoButton(btns); - var currentLasso = makeLassoFunction(n, colorFunction$$1.colors()[cN]); - var clearBtn = makeClearButton(btns); + var path; - bindButtons(lassoBtn, clearBtn, currentLasso, table); + lasso.svg = function(_) { return arguments.length ? (svg = _, lasso) : svg; }; + lasso.chartContainer = function(_) { return arguments.length ? (chartContainer = _, lasso) : chartContainer; }; + lasso.objectContainer = function(_) { return arguments.length ? (objectContainer = _, lasso) : objectContainer; }; + lasso.objectClass = function(_) { return arguments.length ? (objectClass = _, lasso) : objectClass; }; + lasso.namespace = function(_) { return arguments.length ? (namespace = _, lasso) : namespace; }; + lasso.xScale = function(_) { return arguments.length ? (xScale = _, lasso) : xScale; }; + lasso.yScale = function(_) { return arguments.length ? (yScale = _, lasso) : yScale; }; + lasso.activeQ = function(_) { return arguments.length ? (activeQ = _, lasso) : activeQ; }; + lasso.currentPoints = function(_) { return arguments.length ? (currentPoints = _, lasso) : currentPoints; }; + lasso.allPoints = function(_) { return arguments.length ? (allPoints = _, lasso) : allPoints; }; + lasso.instance = function(_) { return arguments.length ? (instance = _, lasso) : instance; }; + lasso.tickDistance = function(_) { return arguments.length ? (tickDistance = _, lasso) : tickDistance; }; + lasso.color = function(_) { return arguments.length ? (color$$1 = _, lasso) : color$$1; }; + lasso.animationRate = function(_) { return arguments.length ? (animationRate = _, lasso) : animationRate; }; + lasso.opacity = function(_) { return arguments.length ? (opacity = _, lasso) : opacity; }; + lasso.dashArray = function(_) { return arguments.length ? (dashArray = _, lasso) : dashArray; }; + lasso.stroke = function(_) { return arguments.length ? (stroke = _, lasso) : stroke; }; + lasso.lassoedFill = function(_) { return arguments.length ? (lassoedFill = _, lasso) : lassoedFill; }; + lasso.lassoedStroke = function(_) { return arguments.length ? (lassoedStroke = _, lasso) : lassoedStroke; }; + lasso.lassoedStrokeWidth = function(_) { return arguments.length ? (lassoedStrokeWidth = _, lasso) : lassoedStrokeWidth; }; + lasso.eventCatcher = function(_) { return arguments.length ? (eventCatcher = _, lasso) : eventCatcher; }; - var removeBtn = makeRemoveButton(btns); + lasso.drag = drag; + lasso.draw = draw; + lasso.tick = tick; + lasso.detect = detect; + lasso.toggle = toggle; + lasso.remove = remove; + lasso.render = render; + lasso.keyFrames = keyFrames; + lasso.updateObjects = updateObjects; + lasso.applyPathAttributes = applyPathAttributes; + lasso.applyObjectAttributes = applyObjectAttributes; - lead.on("mouseover", function(){ - currentLasso.draw(); - }); - lead.on("mouseout", function(){ - currentLasso.remove(); - }); + keyFrames(); + function lasso() { + // add a dash animation if needed + if (activeQ) { transitionDraw(); } } - function makeLassoButton(btns) { - var btn = safeSelect(btns, 'button', 'lasso-btn') - .classed('btn btn-info', true) - .classed(namespace, true); - // .datum([lasso]) - var i = safeSelect(btn, 'i', 'fa').classed('fa-hand-pointer-o', true); - var s = safeSelect(btn, 'span').text('Lasso select'); - return btn - - } + function toggle(state) { + // use optional param to set state, otherwise toggle state + activeQ = (state!=undefined) ? state : !activeQ; + chartOffset = utils$1.sel.getTranslation(chartContainer); + objectsOffset = utils$1.sel.getTranslation(objectContainer); - function makeClearButton(btns) { - var btn = safeSelect(btns, 'button', 'clear-btn') - .classed('btn btn-light', true) - .classed(namespace, true); - var i = safeSelect(btn, 'i', 'fa').classed('fa-eraser', true); - var s = safeSelect(btn, 'span').text('Clear selection'); - return btn - } + if (activeQ) { + svg.node().addEventListener('mousedown', render, true); + } else { + svg.node().removeEventListener('mousedown', render, true); + remove(); + } - function makeLassoFunction(instance, color) { - var currentLasso = lasso() - .namespace(namespace) - .svg(svg) - .objectClass(objectClass) - .chartContainer(chartContainer) - .objectContainer(objectContainer) - .instance(instance) - .color(color) - .yScale(yScale) - .xScale(xScale); - - return currentLasso } - function bindButtons(lassoBtn, clearBtn, currentLasso, table) { - currentLasso.eventCatcher(lassoBtn); - lassoBtn.node().addEventListener("click", lassoBtnToggle); - - lassoBtn.datum([currentLasso]) - .on(hypenate(namespace,'render'), function(){ - d3.select(this).datum()[0].draw(); - }) - .on(hypenate(namespace,"drag"), function(las){ - var nodes = chartContainer.selectAll(".in-lasso-"+las[0].instance()).nodes(); - var tableData = nodes.map(function(d, i){ - var extracted = dataExtractor(d3.select(d).datum()); - extracted["__node"] = d; - return extracted - }); + function draw() { + chartOffset = utils$1.sel.getTranslation(chartContainer); + objectsOffset = utils$1.sel.getTranslation(objectContainer); + var container = utils$1.sel.safeSelect(objectContainer, 'g', 'lasso-container'); + var paths$$1 = container.selectAll('path[instance="'+instance+'"]'); - makeTable(table, tableData, currentLasso); - }); + // update + paths$$1 = paths$$1.data(allPoints); - clearBtn.on('click', function(){ - currentLasso.allPoints([]); - currentLasso.currentPoints([]); - lassoBtn.dispatch(hypenate(namespace,"drag")); + // remove excess + var pExit = paths$$1.exit().remove(); + // add needed paths + var pEnter = paths$$1.enter().append('path'); - }); + // merge + paths$$1 = paths$$1.merge(pEnter); + // apply + applyPathAttributes(paths$$1); } - function lassoBtnToggle() { - var that = d3.select(this); - var las = that.datum()[0]; - - las.toggle(); - var activeQ = las.activeQ(); - - if (las.activeQ()) { - that.classed('btn-info', !activeQ); - that.classed('btn-warning', activeQ); - that.select("span").text("Lasso select (active)"); - selection.selectAll("."+namespace+".lasso-btn").dispatch(hypenate(namespace,'render')); - d3.select("html").node().addEventListener('mousedown', monitorLassoButtonState); - } else { - that.classed('btn-info', !activeQ); - that.classed('btn-warning', activeQ); - that.select("span").text("Lasso select"); - d3.select("html").node().removeEventListener('mousedown', monitorLassoButtonState); - } - - function monitorLassoButtonState(event) { - /* - activeLasso stops event stopPropagation, so this event will not register in - that case. Thus we only need to ensure that the usre is clicking on the lasso button - otherwise, de-spawn lasso. - */ - if ( - event.target != that.node() && - event.target != that.select("span").node() && - event.target != that.select("i").node() - ) { - // event.preventDefault() - // event.stopPropagation() - las.toggle(false); - that.classed('btn-info', true); - that.classed('btn-warning', false); - that.select("span").text("Lasso select"); - // console.log(that, that.node()) - // that.dispatch("focusout") - } - } - + function remove() { + var container = utils$1.sel.safeSelect(objectContainer, 'g', 'lasso-container'); + var paths$$1 = container.selectAll('path[instance="'+instance+'"]').remove(); + container.remove(); + objectContainer.selectAll(objectClass).classed("in-lasso", false); + updateObjects(); } + function render( event ) { + // nothing can interefer with drawing the lasso + event.preventDefault(); event.stopPropagation(); - function updateTableHeaderColumns(headR, tableData) { - var headerKeys = d3.keys(tableData[0]).filter(k=>k!="__node"); - if (headerKeys.length > 0) { - // headerKeys = ["remove"].concat(headerKeys) - headerKeys.push("remove"); - } - - var headerCols = headR.selectAll("th"); - - headerCols = headerCols.data(headerKeys); - headerCols.exit().remove(); - headerCols = headerCols.merge(headerCols.enter().append("th").attr("scope","col")) - .text(function(d, i){return d}); - - return headerKeys - } + var container = utils$1.sel.safeSelect(objectContainer, 'g', 'lasso-container'); - function updateTableRows(body, tableData) { - var bodyRows = body.selectAll("tr"); + /* + each time the user presses down, while the state is active, the lasso + the lasso should make a seperate segment. + */ + currentPoints = []; - bodyRows = bodyRows.data(tableData); - bodyRows.exit().remove(); - bodyRows = bodyRows.merge(bodyRows.enter().append("tr")); + svg.node().addEventListener('mousemove', drag); + svg.node().addEventListener('mouseup', function(event) { + svg.node().removeEventListener('mousemove', drag); + allPoints.push(currentPoints); + // BUG: somehow this is pushing currentPoints n times where n is the nth lasso path for the current instance + // NOTE: allPoints = utils.arr.unique(allPoints) is a temporary and inefficient fix + allPoints = utils$1.arr.unique(allPoints); + }); - return bodyRows + path = container.append('path').data([currentPoints]); + applyPathAttributes(path); } - function updateTableRowColumns(cols, headerKeys, rowData, tableData, lasso$$1, table) { - cols = cols.data(headerKeys); - cols.exit().remove(); - cols = cols.merge(cols.enter().append("td")); - - cols.html(function(d, i){ - if (d != "remove") { return rowData[d] } - return "" - }).classed("text-left", function(d, i){ - if (d != "remove") { return false } - return true - }); - // .attr("scope",function(d, i){ - // if (d != "remove") { return false } - // return true - // }) + function transitionDraw() { + var container = utils$1.sel.safeSelect(objectContainer, 'g', 'lasso-container'); + var paths$$1 = container.selectAll('path[instance="'+instance+'"]'); - cols.select("i.fa-close").on("click", function(d, i){ - var node = rowData["__node"], - n = d3.select(node); - n.classed("in-lasso", false); - n.classed("in-lasso-"+lasso$$1.instance(), false); + // update + paths$$1 = paths$$1.data(allPoints); - tableData.map(function(dd, j){ - if (dd["__node"] == node) { - tableData.splice(j, 1); - makeTable(table, tableData, lasso$$1); - } - }); + // remove excess + var pExit = paths$$1.exit().remove(); + // add needed paths + var pEnter = paths$$1.enter().append('path'); + + // merge + paths$$1 = paths$$1.merge(pEnter) + .transition().duration(transitionDuration) + .ease(easeFunc); + applyPathAttributes(paths$$1); - }); } - function makeTable(table, tableData, lasso$$1) { - var - head = safeSelect(table, "thead"), - headR = safeSelect(head, "tr"), - body = safeSelect(table, "tbody"), + function applyPathAttributes(path) { + path + .attr("class", utils$1.str.hypenate(namespace, "lasso-path")) + .style('opacity', opacity) + .attr('fill', color$$1) + .attr("d", line) + .attr('instance', instance) + .style("stroke-dasharray", dashArray) + .attr("stroke", stroke) + .attr("stroke-width", strokeWidth) + .style('animation', 'lassoDash '+animationRate+' linear') + .style("animation-iteration-count", "infinite"); + } - headerKeys = updateTableHeaderColumns(headR, tableData), + function drag(event) { + /* + effectively create a mouse down and move event (which normally is inteperated + as 'drag' by the browser) by dynamically adding / removing this event on + mouse down / mouse up. + */ - bodyRows = updateTableRows(body, tableData); + if (eventCatcher != undefined) {eventCatcher.dispatch(utils$1.str.hypenate(namespace,"drag"));} + // d3.dispatch(utils.str.hypenate(namespace,"drag")) - bodyRows.each(function(rowData, i){ - var t = d3.select(this); - var cols = t.selectAll("td"); + if (event.which != 1) {return} // ensures left mouse button set + d3.event = event; + var pt = d3.mouse(objectContainer.node()); + var pt = d3.mouse(svg.node()); - updateTableRowColumns(cols, headerKeys, rowData, tableData, lasso$$1, table); + if (xScale != undefined) {pt[0] = xScale.invert(pt[0]);} + if (yScale != undefined) {pt[1] = yScale.invert(pt[1]);} + pt[0] = pt[0] - chartOffset[0] - objectsOffset[0]; + pt[1] = pt[1] - chartOffset[1] - objectsOffset[1]; - t.on("mouseover", function(d, i) { - lasso$$1.applyObjectAttributes(d3.select(d["__node"]),true); - }) - .on("mouseout", function(d, i) { - lasso$$1.applyObjectAttributes(d3.select(d["__node"]),false); - }); - }); + /* if we have a point already, test if it passes a minimum distance to prevent overwhelming with too many tick functions */ + if (currentPoints.length) { + var lastPt = currentPoints[currentPoints.length - 1]; + var a = [pt[0], pt[1]], b = [lastPt[0], lastPt[1]]; + if (xScale) {b[0] = xScale(b[0]); a[0] = xScale(a[0]);} + if (yScale) {b[1] = yScale(b[1]); a[1] = yScale(a[1]);} + var dist = utils$1.math.euclideanDistance(b, a); + if (dist > tickDistance) { tick(pt); } + } + else { tick(pt); } } + function tick (pt) { + /* + If a point is provided update data and objects. + Otherwise just call on data we already have. + Why like this?: + 1. currentPoints is current points to allow disjunct lassos, currentPoints is only pushed to + allPoints after mouseup. + 2. to allow render of objects in the lasso class / updating the data list + just by toggling the button + */ + if (pt != undefined) { + currentPoints.push(pt); + path.attr("d", line); + if (currentPoints.length < 3) {return} // need at least 3 points to detect anything. + detect(allPoints.concat([currentPoints])); + } else { + detect(allPoints); + } + } + function detect(lassos) { + if (lassos == undefined) {lassos = allPoints;} + objectContainer.selectAll(objectClass).each(function(d, i){ + var current = d3.select(this), - function makeNewGroup(d, i) { - - var tabs = selection.select('.'+hypenate(namespace, 'tab-list')), - panes = selection.select('.'+hypenate(namespace, 'tab-content')), - n = newGroupNumber(); - - if (tabs.selectAll('.'+hypenate(namespace,'tab')).size() == maxNumberOfGroups) { - onError('only '+maxNumberOfGroups+' allowed.', 'warning'); - return - } + box = current.absolutePosition(), + // box = current.relativePositionTo(objectContainer.node()), - var - tab = insertTab(tabs, n), - pane = makeTabPane(panes, n); + boxPts = [ + [ + box.left - chartOffset[0] - objectsOffset[0], + box.top - chartOffset[1] - objectsOffset[1] + ], + [ + box.right - chartOffset[0] - objectsOffset[0], + box.top - chartOffset[1] - objectsOffset[1] + ], + [ + box.left - chartOffset[0] - objectsOffset[0], + box.bottom - chartOffset[1] - objectsOffset[1] + ], + [ + box.right - chartOffset[0] - objectsOffset[0], + box.bottom - chartOffset[1] - objectsOffset[1] + ] + ]; - populatePane(pane, n); - togglePaneById(pane.attr('id')); + if (xScale != undefined) { + boxPts[0][0] = xScale.invert(boxPts[0][0]); + boxPts[1][0] = xScale.invert(boxPts[1][0]); + boxPts[2][0] = xScale.invert(boxPts[2][0]); + boxPts[3][0] = xScale.invert(boxPts[3][0]); + } + if (yScale != undefined) { + boxPts[0][1] = yScale.invert(boxPts[0][1]); + boxPts[1][1] = yScale.invert(boxPts[1][1]); + boxPts[2][1] = yScale.invert(boxPts[2][1]); + boxPts[3][1] = yScale.invert(boxPts[3][1]); + } - } - function newGroupNumber() { - var tabs = selection.select('.'+hypenate(namespace, 'tab-list'));//, - var - n = tabs.selectAll('li').size() - 3, // minus 1 for plus tab - s = 'Group ' + n, - gs = []; - tabs.each(function(d, i){ return gs.push(d3.select(this).select("a").text()) }); - while (gs.includes(s)) { n+=1; s = 'Group ' + n; } - return n - } + /* + flag needed as we have to test multiple lasso segments, and if the point + is not in one segment, it does not mean it is not in any + */ + var inAnyLassoQ = false; + for (var i = 0; i < lassos.length; i++) { + var lassoPoints = lassos[i]; + // .map(function(pt){ + // var x, y = pt + // if (xScale!=undefined) {x = xScale(x)} + // if (yScale!=undefined) {y = yScale(y)} + // return [x, y] + // }) + var boxInLassoQ = boxPts.every(coord => d3.polygonContains(lassoPoints, coord)); - function updateTabAndPaneNumbers() { - var - tabs = selection.select('.'+hypenate(namespace, 'tab-list')), - panes = selection.select('.'+hypenate(namespace, 'tab-content')); - - tabs = tabs.selectAll('.'+hypenate(namespace,'tab')); - panes = panes.selectAll('.'+hypenate(namespace,'tab', 'pane')); - - tabs.each(function(d, i){ - d3.select(this).datum(i) - .attr("id", hypenate(namespace,'tab',i)) - .select('a') - .attr('href', '#'+hypenate(namespace,'tab','pane',i)) - .text(function(dd, ii){ - var curText = d3.select(this).text(); - if (curText.split(' ')[0] == 'Group') { - return 'Group ' + i - } - return curText - }); - }); + if (boxInLassoQ) { inAnyLassoQ = true; } // only update flag in the positive case. + } - panes.each(function(d, i){ - d3.select(this).datum(i) - .attr('id', hypenate(namespace,'tab','pane',i)); - safeSelect(d3.select(this), 'p', 'lead') - .text(function(dd, ii){ - var curText = d3.select(this).text(); - if (curText.split(' ')[0] == 'Group') { - return 'Group ' + i - } - return curText - }); - safeSelect(d3.select(this), 'button', 'remove-btn') - .on('click', removeBtnClickToRemove); + current.classed('in-lasso', inAnyLassoQ); + current.classed('in-lasso-'+instance, inAnyLassoQ); }); + updateObjects(); + return objectContainer.selectAll('.in-lasso-'+instance) } - function gatherDataLists() { - var tabs = selection.select('.'+hypenate(namespace, 'tab-list')); - var panes = selection.select('.'+hypenate(namespace, 'tab-content')); - var tables = panes.selectAll('.'+hypenate(namespace, "data-table")); - var data = {}; - var textGroups = tabs.selectAll('li.'+hypenate(namespace,'tab') + ' > a') - .nodes().map(function(d, i){return d3.select(d).text()}); - textGroups.map(function(e, i){ - data[e] = []; + function updateObjects() { + objectContainer.selectAll(objectClass).each(function(d, i) { + var t = d3.select(this); + applyObjectAttributes(t, t.classed('in-lasso')); }); + } + function applyObjectAttributes(obj, setQ) { + var + preLassoFill = obj.attr('_pre_lasso_fill'), + preLassoStroke = obj.attr('_pre_lasso_stroke'), + preLassoStrokeWidth = obj.attr('_pre_lasso_stroke-width'); - tables.each(function(d, i){ - var table = d3.select(this).select("tbody"); - data[textGroups[i]] = table.selectAll('tr').data(); - }); + if (setQ) { + obj.classed("in-lasso", true); + obj.classed('in-lasso-'+instance, true); + if (preLassoFill == undefined) { obj.attr('_pre_lasso_fill', obj.attr('fill')); } + if (preLassoStroke == undefined) { obj.attr('_pre_lasso_stroke', obj.attr('stroke')); } + if (preLassoStrokeWidth == undefined) { obj.attr('_pre_lasso_stroke-width', obj.attr('stroke-width')); } - return data - } + obj + //BUG: when .raise() + .attr('fill', lassoedFill) + .attr('stroke', lassoedStroke) + .attr('stoke-width', lassoedStrokeWidth); - function showRemainingPanes() { - var - tabs = selection.select('.'+hypenate(namespace, 'tab-list')), - panes = selection.select('.'+hypenate(namespace, 'tab-content')), - remainingTabs = tabs.selectAll('.'+hypenate(namespace,'tab')); - - if (remainingTabs.size() == 0) { - panes.select('.'+hypenate(namespace,'default-tab')) - .classed("active", true) - .classed('text-left', true); - } - else { - var lastTab = remainingTabs.nodes()[remainingTabs.size()-1], - lastPaneId = d3.select(lastTab).select('a').attr('href'); - panes.select(lastPaneId) - .classed("active", true) - .classed('text-left', true); + } else { + obj.classed("in-lasso", false); + obj.classed('in-lasso-'+instance, false); + if (preLassoFill != undefined) { obj.attr('fill', preLassoFill); } + if (preLassoStroke != undefined) { obj.attr('stroke', preLassoStroke); } + if (preLassoStrokeWidth != undefined) { obj.attr('stroke-width', preLassoStrokeWidth); } } } + function keyFrames() { + var style = + d3.select("html").select('style.'+utils$1.str.hypenate(namespace,"lasso-dash")); + if (style.empty()) { + d3.select("html").append('style') + .classed(utils$1.str.hypenate(namespace,"lasso-dash"), true) + .html("@keyframes lassoDash {to { stroke-dashoffset: 1000;}}"); + } - return lassoWidget + } + return lasso } -function upset ( selection ) { - var - data, - orient='horizontal', - spaceX, - spaceY, - overflowQ=false, - minObjectSize=20, - maxObjectSize=50, - circleStrokeWidth=2, - // colorFunction= - backgroundFill = 'transparent', - namespace='d3sm-upset', - objectClass = 'upset', - - transitionDuration = 1000, - easeFunc = d3.easeExp, - - setKey = "set", - intersectionKey = "intersection", - elementsKey = "elements", - - setExtractor = function(key, i) {return data[key][setKey]}, - intersectionExtractor = function(key, i) {return data[key][intersectionKey]}, - elementExtractor = function(key, i) {return data[key][elementsKey]}, - - cellKeys, - setValues, - intersectionValues, - xObjectSpacer = 0.05, - yObjectSpacer = 0.05, - radius, +/******************************************************************************* - // listDelim = ';' +** ** +** ** +** PLOTZOOM ** +** ** +** ** +*******************************************************************************/ +/** + * Creates an plotZoom instance, which can handle drag and scroll events + * @constructor plotZoom + * @param {function} chart a function instance of one of the d3sm plots (e.g. bar, boxwhisker, bubbleHeatmap, violin, etc) + * @param {axis} xAxis the axis instance responsible for the x axis + * @param {axis} yAxis the axis instance responsible for the y axis + * @namespace plotZoom + * @returns {function} zoom + */ +function multiPlotZoom( chart ) { + var + /** + * The event on which to fire + * (see {@link plotZoom#eventType}) + * @param {string} eventType which event it should handle. Currently supports scroll and wheel + * @memberof plotZoom# + * @property + */ + eventType, + /** + * A scaling factor for the wheel "speed" + * (see {@link plotZoom#wheelSpeed}) + * @param {number} [wheelSpeed=20] scales the wheel translation by wheelSpeed + * @memberof plotZoom# + * @property + */ + wheelSpeed = 20, + /** + * The orientation in which to allow scrolling: 'horizontal', 'vertical', or '2D' + * (see {@link plotZoom#orient}) + * @param {string} [orient=chart.orient() || 'horizontal'] + * @memberof plotZoom# + * @property + */ + orient = (chart.orient == undefined) ? 'horizontal' : chart.orient(), + /** + * The max distance allowed to scroll in the x direction + * (see {@link plotZoom#xLock}) + * @param {number} [xLock=chart.spaceX()] ideally chart.overflowQ() == true and this value is the + * bounding rect across all elements in the chart minus the space in which to show. + * @memberof plotZoom# + * @property + */ + xLock=chart.spaceX(), + /** + * The max distance allowed to scroll in the y direction + * (see {@link plotZoom#yLock}) + * @param {number} [yLock=chart.spaceY()] ideally chart.overflowQ() == true and this value is the + * bounding rect across all elements in the chart minus the space in which to show. + * @memberof plotZoom# + * @property + */ + yLock=chart.spaceY(), - yObjectSize, - ySpacerSize, - xObjectSize, - xSpacerSize, + chartSel = chart.selection(), - setKeySortingFunction = function(a, b) { return setValues.indexOf(setExtractor(a)) - setValues.indexOf(setExtractor(b)) }, - intersectionKeySortingFunction = function(a, b) { return intersectionValues.indexOf(intersectionExtractor(a)) - intersectionValues.indexOf(intersectionExtractor(b)) }; + svg = d3.select(chartSel.thisSVG()), + xComponents = [], + yComponents = []; - upset.selection = function(_) { return arguments.length ? (selection = _, upset) : selection; }; - upset.data = function(_) { return arguments.length ? (data = _, upset) : data; }; - upset.orient = function(_) { return arguments.length ? (orient = _, upset) : orient; }; - upset.spaceX = function(_) { return arguments.length ? (spaceX = _, upset) : spaceX; }; - upset.spaceY = function(_) { return arguments.length ? (spaceY = _, upset) : spaceY; }; - upset.overflowQ = function(_) { return arguments.length ? (overflowQ = _, upset) : overflowQ; }; - upset.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, upset) : minObjectSize; }; - upset.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, upset) : maxObjectSize; }; - upset.circleStrokeWidth = function(_) { return arguments.length ? (circleStrokeWidth = _, upset) : circleStrokeWidth; }; - upset.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, upset) : backgroundFill; }; - upset.namespace = function(_) { return arguments.length ? (namespace = _, upset) : namespace; }; - upset.objectClass = function(_) { return arguments.length ? (objectClass = _, upset) : objectClass; }; - upset.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, upset) : transitionDuration; }; - upset.easeFunc = function(_) { return arguments.length ? (easeFunc = _, upset) : easeFunc; }; - upset.cellKeys = function(_) { return arguments.length ? (cellKeys = _, upset) : cellKeys; }; - upset.setValues = function(_) { return arguments.length ? (setValues = _, upset) : setValues; }; - upset.intersectionValues = function(_) { return arguments.length ? (intersectionValues = _, upset) : intersectionValues; }; - upset.xObjectSpacer = function(_) { return arguments.length ? (xObjectSpacer = _, upset) : xObjectSpacer; }; - upset.yObjectSpacer = function(_) { return arguments.length ? (yObjectSpacer = _, upset) : yObjectSpacer; }; - upset.radius = function(_) { return arguments.length ? (radius = _, upset) : radius; }; - upset.setExtractor = function(_) { return arguments.length ? (setExtractor = _, upset) : setExtractor; }; - upset.intersectionExtractor = function(_) { return arguments.length ? (intersectionExtractor = _, upset) : intersectionExtractor; }; - upset.elementExtractor = function(_) { return arguments.length ? (elementExtractor = _, upset) : elementExtractor; }; - upset.setKeySortingFunction = function(_) { return arguments.length ? (setKeySortingFunction = _, upset) : setKeySortingFunction; }; - upset.intersectionKeySortingFunction = function(_) { return arguments.length ? (intersectionKeySortingFunction = _, upset) : intersectionKeySortingFunction; }; - upset.yObjectSize = function(_) { return arguments.length ? (yObjectSize = _, upset) : yObjectSize; }; - upset.ySpacerSize = function(_) { return arguments.length ? (ySpacerSize = _, upset) : ySpacerSize; }; - upset.xObjectSize = function(_) { return arguments.length ? (xObjectSize = _, upset) : xObjectSize; }; - upset.xSpacerSize = function(_) { return arguments.length ? (xSpacerSize = _, upset) : xSpacerSize; }; + /** + * Gets or sets the event type in which to respond + * (see {@link plotZoom#eventType}) + * @param {string} [_=none] should be 'drag' or 'wheel' + * @returns {zoom | string} + * @memberof plotZoom + * @property + * by default plotZoom=undefined + */ + zoom.eventType = function(_) { return arguments.length ? (eventType = _, zoom) : eventType; }; + /** + * Gets or sets the wheel speed in which to scale the wheel scroll transform + * (see {@link plotZoom#wheelSpeed}) + * @param {number} [_=none] + * @returns {zoom | number} + * @memberof plotZoom + * @property + * by default wheelSpeed=20 + */ + zoom.wheelSpeed = function(_) { return arguments.length ? (wheelSpeed = _, zoom) : wheelSpeed; }; + /** + * Gets or sets the orientation in which items are manipulated + * (see {@link plotZoom#orient}) + * @param {string} [_=none] should be horizontal, vertical, or 2D + * @returns {zoom | string} + * @memberof plotZoom + * @property + * by default orient=chart.orient() || 'horizontal' + */ + zoom.orient = function(_) { return arguments.length ? (orient = _, zoom) : orient; }; - function upset() { - // for convenience in handling orientation specific values - var horizontalQ = (orient == 'horizontal') ? true : false; - var verticalQ = !horizontalQ; + /** + * Gets or sets the max distance in which to scroll X + * (see {@link plotZoom#xLock}) + * @param {number} [_=none] should be a positive value + * @returns {zoom | number} + * @memberof plotZoom + * @property + * by default xLock=chart.spaceX() + */ + zoom.xLock = function(_) { return arguments.length ? (xLock = _, zoom) : xLock; }; + /** + * Gets or sets the max distance in which to scroll Y + * (see {@link plotZoom#yLock}) + * @param {number} [_=none] should be a positive value + * @returns {zoom | number} + * @memberof plotZoom + * @property + * by default yLock=chart.spaceY() + */ + zoom.yLock = function(_) { return arguments.length ? (yLock = _, zoom) : yLock; }; - // background cliping rectangle - var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; - var container = setupContainer( selection, namespace, bgcpRect, backgroundFill ); + zoom.xComponents = function(_) { return arguments.length ? (xComponents = _, zoom) : xComponents; }; + zoom.yComponents = function(_) { return arguments.length ? (yComponents = _, zoom) : yComponents; }; - cellKeys = d3.keys(data); - setValues = unique(cellKeys.map(setExtractor)).sort(); - intersectionValues = unique(cellKeys.map(intersectionExtractor)).sort().sort(function(a, b){ - return a.split(';').length - b.split(';').length - }); + function setLocks() { + var chartObjSel = chartSel.select('.'+utils$1.str.hypenate(chart.namespace(),'object-container')); + var chartObjTrans = utils$1.sel.getTranslation(chartObjSel.attr('transform')); + var cos = chartObjSel.attr('transform', 'translate(0,0)'); - if (!horizontalQ) { - cellKeys.sort(function(a, b){ return setKeySortingFunction(a, b) || intersectionKeySortingFunction(a, b) }); - } else { - cellKeys.sort(function(a, b){ return intersectionKeySortingFunction(a, b) || setKeySortingFunction(a, b) }); - } + xLock = chartSel.node().getBBox().width - chart.spaceX();// * .9 + yLock = chartSel.node().getBBox().height - chart.spaceY();// * .9 + cos.attr('transform', 'translate('+chartObjTrans[0]+','+chartObjTrans[1]+')'); + utils$1.con.log('plotZoom', 'setLocks', {xLock:xLock, yLock:yLock}); + } + /** + * Sets the x and y locks (how far one can scroll) + * (see {@link plotZoom#xLock} and {@link plotZoom#yLock}) + * @function plotZoom.setLocks + * @returns {undefined} + * @memberof plotZoom + * @property + */ + zoom.setLocks = setLocks; + function zoom() { + setLocks(); var - xValues = horizontalQ ? intersectionValues : setValues, - yValues = horizontalQ ? setValues : intersectionValues, - xDim = horizontalQ ? xValues.length : yValues.length, - yDim = horizontalQ ? yValues.length : xValues.length; - - // console.log(xValues, yValues) - - - xObjectSize = calculateWidthOfObject(spaceX, xDim, minObjectSize, maxObjectSize, xObjectSpacer, overflowQ); - yObjectSize = calculateWidthOfObject(spaceY, yDim, minObjectSize, maxObjectSize, yObjectSpacer, overflowQ); - xSpacerSize = calculateWidthOfSpacer(xValues, spaceX, xObjectSize, xDim, xObjectSpacer, overflowQ); - ySpacerSize = calculateWidthOfSpacer(yValues, spaceY, yObjectSize, yDim, yObjectSpacer, overflowQ); + xComponentsSel = xComponents.map(function(d, i){return d.selection()}), + yComponentsSel = yComponents.map(function(d, i){return d.selection()}); - var ySpacer = groupingSpacer() - .horizontalQ(false) - .moveby('category').numberOfObjects(yDim) - .objectSize(yObjectSize).spacerSize(ySpacerSize) - .transitionDuration(transitionDuration).easeFunc(easeFunc); + var horizontalQ, verticalQ; + if (orient == '2D') {horizontalQ = true; verticalQ = true;} + if (orient == 'horizontal') {horizontalQ = true; verticalQ = false;} + if (orient == 'vertical') {verticalQ = true; horizontalQ = false;} - var xSpacer = groupingSpacer() - .horizontalQ(true) - .moveby('category').numberOfObjects(xDim) - .objectClass(objectClass) - .objectSize(xObjectSize).spacerSize(xSpacerSize) - .transitionDuration(transitionDuration).easeFunc(easeFunc); + // capture transform event + var transform = d3.event.transform; + var chartBox = chartSel.node().getBBox(); + var xComponentsBox = xComponentsSel.map(function(d, i){return d.node().getBBox()}); + var yComponentsBox = xComponentsSel.map(function(d, i){return d.node().getBBox()}); - if (verticalQ) { - xSpacer.objectClass(objectClass); - ySpacer.namespace('across').objectClass(hypenate(objectClass, 'across')); + var chartWidth = chartBox.width - chartBox.x; + var chartHeight = chartBox.height - chartBox.y; - ySpacer(container, yValues, 0); - container.selectAll('g.'+hypenate(objectClass, 'across')) - .each(function(d, i){ xSpacer(d3.select(this), xValues, 0); }); - } else { - xSpacer.namespace('across').objectClass(hypenate(objectClass, 'across')); - ySpacer.objectClass(objectClass); + // enable wheel event + if (eventType == "wheel") { + var e = d3.event; + // prevent page scrolling + e.preventDefault(); + // event delta is very very slow, so speed it up with wheel speed + var w = d3.event.deltaY * wheelSpeed; + var shiftQ = d3.event.shiftKey; - xSpacer(container, xValues, 0); - container.selectAll('g.'+hypenate(objectClass, 'across')) - .each(function(d, i){ ySpacer(d3.select(this), yValues, 0); }); + // d3 has no way to make custom transform, so make an object and add + // the necessary functions to make wheel event compatible with drag events + if (orient == '2D') { + transform = shiftQ ? {k: 1, x: w, y: 0} : {k: 1, x: 0, y: w}; + } else { + transform = horizontalQ ? {k: 1, x: w, y: 0} : {k: 1, x: 0, y: w}; + } + // * -1 is invert + transform.applyX = function(x) { return x * this.k + this.x *-1; }; + transform.applyY = function(y) { return y * this.k + this.y *-1; }; } - var cells = container.selectAll('g:not(.to-remove).'+objectClass); - var lookup = {}; - cellKeys.map(function(k, i){ - lookup[setExtractor(k)+'::'+intersectionExtractor(k)] = k; - }); - - // var positionedCellKeys = [] - // for (var i = 0; i < setValues.length; i++) { - // for (var j = 0; j < intersectionValues.length; j++) { - // var lookupValue = lookup[setValues[j]+"::"+intersectionValues[i]] - // if (lookupValue == undefined) { - // positionedCellKeys.push(undefined) - // } else { - // positionedCellKeys.push(lookupValue) - // } - // console.log(i, j, lookupValue) - // } - // } - // - // // console.log(positionedCellKeys) - // - // cells.data(positionedCellKeys); - - - cells.data(cellKeys); - cells.each(function(key, i) { - var t = d3.select(this); - if (key == undefined) {return } - var - currentData = data[key]; - // console.log(key, currentData) - var - set = setExtractor(key, i), - intersection = intersectionExtractor(key, i); + var chartObjSel = chartSel.select('.'+utils$1.str.hypenate(chart.namespace(),'object-container')); + var xComponentObjSel = xComponentsSel.map(function(d, i){ + return d.select('.'+utils$1.str.hypenate(xComponents[i].namespace(),'object-container')) + }); + var yComponentObjSel = yComponentsSel.map(function(d, i){ + return d.select('.'+utils$1.str.hypenate(yComponents[i].namespace(),'object-container')) + }); - // console.log(set, intersection) + var chartObjTrans = utils$1.sel.getTranslation(chartObjSel.attr('transform')); + var xComponentsObjTrans = xComponentObjSel.map(function(d, i){ + return utils$1.sel.getTranslation(d.attr('transform')) + }); + var yComponentsObjTrans = yComponentObjSel.map(function(d, i){ + return utils$1.sel.getTranslation(d.attr('transform')) + }); - t.classed(intersection, true); - t.classed(set, true); + var x = horizontalQ ? transform.applyX(chartObjTrans[0]) : 0; + if (horizontalQ) {x = x < -xLock ? (transform.x = 0, -xLock) : (transform.x = 0, Math.min(x, 0)); } - var c = safeSelect(t, 'circle', hypenate(objectClass,'circle')); - c.attr('cx', xObjectSize / 2) - .attr('cy', yObjectSize / 2 ) - .attr('r', radius == undefined ? Math.min(xObjectSize, yObjectSize) / 2 : radius) - .attr('fill', intersection.includes(set) ? "black": 'rgb(233,233,233)') - .attr('stroke', "black") - .attr("in-intersection", intersection.includes(set)); - }); + var y = verticalQ ? transform.applyY(chartObjTrans[1]) : 0; + if (verticalQ) {y = y < -yLock ? (transform.y = 0, -yLock): (transform.y = 0, Math.min(y, 0));} + chartObjSel.attr('transform', 'translate('+x+','+y+')'); + if (horizontalQ) { + // xAxisObjSel.attr('transform', 'translate('+x+','+0+')') + xComponentObjSel.map(function(d, i){ d.attr('transform', 'translate('+x+','+0+')'); }); + } + if (verticalQ) { + // yAxisObjSel.attr('transform', 'translate('+0+','+y+')') + yComponentObjSel.map(function(d, i){ d.attr('transform', 'translate('+0+','+y+')'); }); + + } } - function intersectionTotals() { - var totals = {}; - // intersectionValues.sort(function(a, b){ return intersectionKeySortingFunction(a, b) }) + zoom.reset = function() { - intersectionValues.map(function(k, i){ totals[k] = {'total': 0}; }); - cellKeys.map(function(k, i){ - var e = elementExtractor(k, i); - if (totals[intersectionExtractor(k, i)]['total'] == 0) { - if (Array.isArray(e)) { - totals[intersectionExtractor(k, i)]['total']+= e.length; - totals[intersectionExtractor(k, i)]['values'] = e; - } else { - totals[intersectionExtractor(k, i)]['total']+= e; - } + var chartObjSel = chartSel.select('.'+utils$1.str.hypenate(chart.namespace(),'object-container')); + var xAxisObjSel = xAxisSel.select('.'+utils$1.str.hypenate(xAxis.namespace(),'object-container')); + var yAxisObjSel = yAxisSel.select('.'+utils$1.str.hypenate(yAxis.namespace(),'object-container')); + chartObjSel.attr('transform', 'translate('+0+','+0+')'); + xAxisObjSel.attr('transform', 'translate('+0+','+0+')'); + yAxisObjSel.attr('transform', 'translate('+0+','+0+')'); + }; - } - }); - return totals - } + return zoom +} - function setTotals(){ - var totals = {}; - // intersectionValues.sort(function(a, b){ return intersectionKeySortingFunction(a, b) }) +/******************************************************************************* - setValues.map(function(k, i){ totals[k] = {'total': 0}; }); +** ** +** ** +** PLOTZOOM ** +** ** +** ** +*******************************************************************************/ +/** + * Creates an plotZoom instance, which can handle drag and scroll events + * @constructor plotZoom + * @param {function} chart a function instance of one of the d3sm plots (e.g. bar, boxwhisker, bubbleHeatmap, violin, etc) + * @param {axis} xAxis the axis instance responsible for the x axis + * @param {axis} yAxis the axis instance responsible for the y axis + * @namespace plotZoom + * @returns {function} zoom + */ +function plotZoom( chart, xAxis, yAxis ) { + var + /** + * The event on which to fire + * (see {@link plotZoom#eventType}) + * @param {string} eventType which event it should handle. Currently supports scroll and wheel + * @memberof plotZoom# + * @property + */ + eventType, + /** + * A scaling factor for the wheel "speed" + * (see {@link plotZoom#wheelSpeed}) + * @param {number} [wheelSpeed=20] scales the wheel translation by wheelSpeed + * @memberof plotZoom# + * @property + */ + wheelSpeed = 20, + /** + * The orientation in which to allow scrolling: 'horizontal', 'vertical', or '2D' + * (see {@link plotZoom#orient}) + * @param {string} [orient=chart.orient() || 'horizontal'] + * @memberof plotZoom# + * @property + */ + orient = (chart.orient == undefined) ? 'horizontal' : chart.orient(), + /** + * The max distance allowed to scroll in the x direction + * (see {@link plotZoom#xLock}) + * @param {number} [xLock=chart.spaceX()] ideally chart.overflowQ() == true and this value is the + * bounding rect across all elements in the chart minus the space in which to show. + * @memberof plotZoom# + * @property + */ + xLock=chart.spaceX(), + /** + * The max distance allowed to scroll in the y direction + * (see {@link plotZoom#yLock}) + * @param {number} [yLock=chart.spaceY()] ideally chart.overflowQ() == true and this value is the + * bounding rect across all elements in the chart minus the space in which to show. + * @memberof plotZoom# + * @property + */ + yLock=chart.spaceY(), - cellKeys.map(function(k, i){ - var e = elementExtractor(k, i); - if (Array.isArray(e)) { - totals[setExtractor(k, i)]['total']+= e.length; - } else { - totals[setExtractor(k, i)]['total']+= e; - } - }); - return totals - } + chartSel = chart.selection(), + xAxisSel = xAxis.selection(), + yAxisSel = yAxis.selection(), + svg = d3.select(chartSel.thisSVG()); - upset.intersectionTotals = intersectionTotals; - upset.setTotals = setTotals; - return upset -} + /** + * Gets or sets the event type in which to respond + * (see {@link plotZoom#eventType}) + * @param {string} [_=none] should be 'drag' or 'wheel' + * @returns {zoom | string} + * @memberof plotZoom + * @property + * by default plotZoom=undefined + */ + zoom.eventType = function(_) { return arguments.length ? (eventType = _, zoom) : eventType; }; + /** + * Gets or sets the wheel speed in which to scale the wheel scroll transform + * (see {@link plotZoom#wheelSpeed}) + * @param {number} [_=none] + * @returns {zoom | number} + * @memberof plotZoom + * @property + * by default wheelSpeed=20 + */ + zoom.wheelSpeed = function(_) { return arguments.length ? (wheelSpeed = _, zoom) : wheelSpeed; }; + /** + * Gets or sets the orientation in which items are manipulated + * (see {@link plotZoom#orient}) + * @param {string} [_=none] should be horizontal, vertical, or 2D + * @returns {zoom | string} + * @memberof plotZoom + * @property + * by default orient=chart.orient() || 'horizontal' + */ + zoom.orient = function(_) { return arguments.length ? (orient = _, zoom) : orient; }; -function filterTable(selection) { - var - data, - namespace = 'd3sm-filter-table', - fieldFunction = function(record, columnKey, recordValue) {return recordValue}; + /** + * Gets or sets the max distance in which to scroll X + * (see {@link plotZoom#xLock}) + * @param {number} [_=none] should be a positive value + * @returns {zoom | number} + * @memberof plotZoom + * @property + * by default xLock=chart.spaceX() + */ + zoom.xLock = function(_) { return arguments.length ? (xLock = _, zoom) : xLock; }; + /** + * Gets or sets the max distance in which to scroll Y + * (see {@link plotZoom#yLock}) + * @param {number} [_=none] should be a positive value + * @returns {zoom | number} + * @memberof plotZoom + * @property + * by default yLock=chart.spaceY() + */ + zoom.yLock = function(_) { return arguments.length ? (yLock = _, zoom) : yLock; }; - filterTable.data = function(_) { return arguments.length ? (data = _, filterTable) : data}; - filterTable.namespace = function(_) { return arguments.length ? (namespace = _, filterTable) : namespace}; - filterTable.fieldFunction = function(_) { return arguments.length ? (fieldFunction = _, filterTable) : fieldFunction}; - function filterTable () { + function setLocks() { + var chartObjSel = chartSel.select('.'+utils$1.str.hypenate(chart.namespace(),'object-container')); + var chartObjTrans = utils$1.sel.getTranslation(chartObjSel.attr('transform')); + var cos = chartObjSel.attr('transform', 'translate(0,0)'); + xLock = chartSel.node().getBBox().width - chart.spaceX() * .9; + yLock = chartSel.node().getBBox().height - chart.spaceY() * .9; + cos.attr('transform', 'translate('+chartObjTrans[0]+','+chartObjTrans[1]+')'); + utils$1.con.log('plotZoom', 'setLocks', {xLock:xLock, yLock:yLock}); + } - var - container = safeSelect(selection, 'div', 'filter-table').classed(hypenate(namespace,'container'),true), - inputGroup = safeSelect(container, 'div', 'filter-input-group').classed('input-group',true), - inputPrepend = safeSelect(inputGroup, 'div', 'input-group-prepend'), - inputPrependSpan = safeSelect(inputPrepend, 'span', 'input-group-text').classed('search-button', true), - inputPrependSpanIcon = safeSelect(inputPrependSpan,'i','fa fa-search'), + /** + * Sets the x and y locks (how far one can scroll) + * (see {@link plotZoom#xLock} and {@link plotZoom#yLock}) + * @function plotZoom.setLocks + * @returns {undefined} + * @memberof plotZoom + * @property + */ + zoom.setLocks = setLocks; - input = safeSelect(inputGroup, 'input', 'form-control').attr('placeholder', 'filter').attr('type', 'text'), - inputAppend = safeSelect(inputGroup, 'div', 'input-group-append'), - inputAppendButton = safeSelect(inputAppend, 'a', 'close-button').classed('btn btn-outline-secondary', true), - inputAppendButtonIcon = safeSelect(inputAppendButton, 'i', 'fa fa-close'), + function zoom() { + setLocks(); + var horizontalQ, verticalQ; + if (orient == '2D') {horizontalQ = true; verticalQ = true;} + if (orient == 'horizontal') {horizontalQ = true; verticalQ = false;} + if (orient == 'vertical') {verticalQ = true; horizontalQ = false;} - closeButton = inputAppendButton, + // capture transform event + var transform = d3.event.transform; - rTable = safeSelect(container, 'div', 'table-responsive'), - table = safeSelect(rTable, 'table', 'table') - .classed('table-hover', true) - .classed('table-bordered', true) - .classed("table-striped", true), - tHead = safeSelect(table, 'thead') - .classed("thead-dark", true), - header = safeSelect(tHead, 'tr'), - tBody = safeSelect(table, 'tbody'); + var chartBox = chartSel.node().getBBox(); + var xAxisBox = xAxisSel.node().getBBox(); + var yAxisBox = xAxisSel.node().getBBox(); + var chartWidth = chartBox.width - chartBox.x; + var chartHeight = chartBox.height - chartBox.y; + var xAxisWidth = xAxisBox.width;// - xAxisBox.x + var xAxisHeight = xAxisBox.height;// - xAxisBox.y + var yAxisWidth = yAxisBox.width;// - yAxisBox.x + var yAxisHeight = yAxisBox.height;// -yAxisBox.y - var - rows = d3.keys(data), - cols = d3.keys(data[rows[0]]); + // enable wheel event + if (eventType == "wheel") { + var e = d3.event; + // prevent page scrolling + e.preventDefault(); + // event delta is very very slow, so speed it up with wheel speed + var w = d3.event.deltaY * wheelSpeed; + var shiftQ = d3.event.shiftKey; - var th = makeColumns(header, cols); - var tr = makeRows(tBody, rows); - var tr = populateRecords(tr, cols); + // d3 has no way to make custom transform, so make an object and add + // the necessary functions to make wheel event compatible with drag events + if (orient == '2D') { + transform = shiftQ ? {k: 1, x: w, y: 0} : {k: 1, x: 0, y: w}; + } else { + transform = horizontalQ ? {k: 1, x: w, y: 0} : {k: 1, x: 0, y: w}; + } + // the * -1 inverts the direction + transform.applyX = function(x) { return x * this.k + this.x * -1; }; + transform.applyY = function(y) { return y * this.k + this.y * -1; }; + } - input.on('input', function(d, i){ - var - val = input.property('value'), - reg = new RegExp(val, 'gi'), - use; - if (val == '') {use = rows;} else { - use = []; - rows.map(function(r, i){ - var row = data[r]; - var fieldJoin = cols.map(function(c, j){ - return fieldFunction(row, c, row[c]) - // return row[c] - }).join(''); - var match = fieldJoin.match(reg); - if (match == null || match.join('') == '') ; - else { use.push(r); } - }); - } + var chartObjSel = chartSel.select('.'+utils$1.str.hypenate(chart.namespace(),'object-container')); + var xAxisObjSel = xAxisSel.select('.'+utils$1.str.hypenate(xAxis.namespace(),'object-container')); + var yAxisObjSel = yAxisSel.select('.'+utils$1.str.hypenate(yAxis.namespace(),'object-container')); - tr = makeRows(tBody, use); - tr = populateRecords(tr, cols); - }); + // xLock = chartSel.node().getBBox().width - chart.spaceX() - chartSel.node().getBBox().x + // yLock = chartSel.node().getBBox().height - chart.spaceY() + // console.table({'xLock':xLock, "yLock":yLock}) + // bhm.selection().node().getBBox().width - bhm.spaceX() - closeButton.on('click', function(d, i){ - input.property('value', '').dispatch('input'); - }); - } + var chartObjTrans = utils$1.sel.getTranslation(chartObjSel.attr('transform')); + var xAxisObjTrans = utils$1.sel.getTranslation(xAxisObjSel.attr('transform')); + var yAxisObjTrans = utils$1.sel.getTranslation(yAxisObjSel.attr('transform')); - function makeColumns(header, cols) { - var th = header.selectAll('th'); - th = th.data(cols); - th.exit().remove(); - th = th.merge(th.enter().append('th')); - th.attr('scope', 'col').text(function(d, i){return d}); - return th - } - function makeRows(body, rows) { - var tr = body.selectAll('tr'); - tr = tr.data(rows); - tr.exit().remove(); - tr = tr.merge(tr.enter().append('tr')); - return tr - } + var x = horizontalQ ? transform.applyX(chartObjTrans[0]) : 0; + if (horizontalQ) {x = x < -xLock ? (transform.x = 0, -xLock) : (transform.x = 0, Math.min(x, 0)); } - function populateRecords(rows, cols) { - rows.each(function(r, i){ - var record = data[r], - t = d3.select(this); + var y = verticalQ ? transform.applyY(chartObjTrans[1]) : 0; + if (verticalQ) {y = y < -yLock ? (transform.y = 0, -yLock): (transform.y = 0, Math.min(y, 0));} - var fields = t.selectAll('td'); - fields = fields.data(cols); - fields.exit().remove(); - fields = fields.merge(fields.enter().append('td')); + chartObjSel.attr('transform', 'translate('+x+','+y+')'); + if (horizontalQ) { xAxisObjSel.attr('transform', 'translate('+x+','+0+')'); } + if (verticalQ) { yAxisObjSel.attr('transform', 'translate('+0+','+y+')'); } - fields.attr('scope', function(f, j){ return j == 0 }) - .html(function(f, j){ return fieldFunction(record, f, record[f]) }); + // var lasso = svg.select(".lasso-container") + // if (!lasso.empty()) { + // lasso.attr('transform', 'translate('+x+','+y+')') + // } - }); - return rows } + zoom.reset = function() { + var chartObjSel = chartSel.select('.'+utils$1.str.hypenate(chart.namespace(),'object-container')); + var xAxisObjSel = xAxisSel.select('.'+utils$1.str.hypenate(xAxis.namespace(),'object-container')); + var yAxisObjSel = yAxisSel.select('.'+utils$1.str.hypenate(yAxis.namespace(),'object-container')); + chartObjSel.attr('transform', 'translate('+0+','+0+')'); + xAxisObjSel.attr('transform', 'translate('+0+','+0+')'); + yAxisObjSel.attr('transform', 'translate('+0+','+0+')'); + }; - return filterTable + return zoom } -// Import styles (automatically inject into ). - -// /** @module d3sm */ -var d3sm$1 = {}; -d3sm$1.axis = axis; -d3sm$1.bar = bar; -d3sm$1.bubbleHeatmap = bubbleHeatmap; -d3sm$1.heatmap = heatmap; -d3sm$1.boxwhisker = boxwhisker; -d3sm$1.colorFunction = colorFunction; -d3sm$1.datatoggle = datatoggle; -d3sm$1.groupingSpacer = groupingSpacer; -d3sm$1.tooltip = tooltip; -d3sm$1.scatter = scatter; -d3sm$1.plotZoom = plotZoom; -d3sm$1.multiPlotZoom = multiPlotZoom; -d3sm$1.violin = violin; -d3sm$1.numericLegend = numericLegend; -d3sm$1.categoricLegend = categoricLegend; -d3sm$1.lasso = lasso; -d3sm$1.lassoWidget = lassoWidget; -d3sm$1.selectFilter = selectFilter; -d3sm$1.upset = upset; -d3sm$1.filterTable = filterTable; - -d3sm$1.uniqueElements = uniqueElements; -d3sm$1.getTranslation = getTranslation; -d3sm$1.modifyHexidecimalColorLuminance = modifyHexidecimalColorLuminance; -d3sm$1.tickRange = tickRange; -d3sm$1.quartiles = quartiles; -d3sm$1.extractViolinValues = extractViolinValues; -d3sm$1.hypenate = hypenate; -d3sm$1.round = round; -d3sm$1.getContainingSVG = getContainingSVG; -d3sm$1.interpolateColors = interpolateColors; -d3sm$1.truncateText = truncateText; -d3sm$1.safeSelect = safeSelect; - -d3sm$1.whichBin = whichBin; -d3sm$1.unique = unique; -d3sm$1.flatten = flatten; - -d3sm$1.setupStandardChartContainers = setupStandardChartContainers; -d3sm$1.log = log; -d3sm$1.warn = warn; -d3sm$1.info = info; -d3sm$1.error = error; -d3sm$1.consoleGroup = consoleGroup; -d3sm$1.consoleGroupEnd = consoleGroupEnd; -d3sm$1.resizeDebounce = resizeDebounce; - -d3sm$1.debugQ = false; - - - -// Import a logger for easier debugging -// import debug from 'debug'; -// const log = debug('app:log'); - -// The logger should only be disabled if we're not in production. -// if (ENV !== 'production') { -// // Enable the logger. -// debug.enable('*'); -// log('Logging is enabled!'); -// -// // Enable LiveReload -// document.write( -// ' + - - + + diff --git a/needs-to-be-bundled/distribution.js b/needs-to-be-bundled/distribution.js deleted file mode 100644 index 781b50d..0000000 --- a/needs-to-be-bundled/distribution.js +++ /dev/null @@ -1,241 +0,0 @@ -function distribution() { - // variables that need to be set externally - var container, - drawingSpace, - colors = ["#2c7bb6", "#00a6ca", "#00ccbc", "#90eb9d", "#ffff8c", "#f9d057", "#f29e2e", "#e76818", "#d7191c"], - data, - vScale = d3.scaleLinear(), // scale for values - nScale = d3.scaleLinear(), // number of elements - vKey, - cKey, - cOpacity = 0.2, - strokeWidth = 2, - cScale = d3.scaleSequential(interpolateColors.apply(this, colors)), - histogram, - curve = d3.curveBasis, - distributionContainer, - distributionPath, - rotation='up', - rangePad = 2 - - // variables set internally - var pointsNeeded, currentNumberOfPoints, - values, yValues, rValues, cValues, - dataExtent, fatal, - verbose = true, - animationTime = 1000, - svg - - var hexademicalOpacity = { - 1.0: "FF", 0.9: 'E6', 0.8: 'CC', - 0.7: "B3", 0.6: "99", 0.5: "80", - 0.4: "66", 0.3: "4D", 0.2: "33", - 0.1: "1A", 0.0: "00" - } - - - - - function plot(){ - fatal = warn(container, 'container', undefined, true, verbose) - if ( fatal ) {return} - fatal = warn(data, 'data', undefined, true, verbose) - if ( fatal ) {return} - fatal = warn(vKey, 'vKey', undefined, true, verbose) - if ( fatal ) {return} - fatal = warn(drawingSpace, 'drawingSpace', undefined, true, verbose) - if ( fatal ) {return} - - - svg = d3.select(container.thisSVG()) - - vValues = data.map( function(d, i) { return parseFloat( d[vKey] ) } ) - histogram = d3.histogram().value( function( d ){ return d[vKey] } )(data) - - nValues = histogram.map( function(bin, i) { return bin.length } ) - - - dataExtent = { - v: { - min: d3.min( vValues ), - max: d3.max( vValues ) - }, - n: { - min: d3.min( nValues ), - max: d3.max( nValues ) - } - } - - vScale.domain([ dataExtent.v.min -rangePad, dataExtent.v.max+rangePad ]) - .range([ 0, drawingSpace.x ]) - - nScale.domain([ 0, dataExtent.n.max ]) - .range([ drawingSpace.y, 0 ]) - - - if (cKey != undefined) { - cValues = data.map( function(d, i) { return parseFloat( d[cKey] ) } ) - cScale.domain([ d3.min(cValues), d3.max(cValues) ]) - } - - - - - histoMin = [] - histoMin.x0 = dataExtent.v.min - histoMin.x1 = dataExtent.v.min - - histoMax = [] - histoMax.x0 = dataExtent.v.max - histoMax.x1 = dataExtent.v.max - - - - line = d3.line() - .x( function(bin, i) { return vScale( d3.median( [bin.x0, bin.x1] ) ) }) - .y( function(bin, i) { return nScale( bin.length ) } ) - .curve(curve) - - - - - distributionContainer = container.select('g.distribution-container') - if (distributionContainer.empty()) { - distributionContainer = container.append('g') - .attr('class','distribution-container') - .data([histogram]) - - distributionContainer.append('path') - .attr('class', 'distribution-path') - } - - distributionPath = distributionContainer.select('path.distribution-path') - distributionPath.attr('d', function(d, i) { - var hist = [histoMin].concat(d).concat([histoMax]) - return line(hist) - }) - .attr('fill', function(d, i){ - if (cKey == undefined) { - var colorI = i % colors.length - var c = colors[colorI] - if (c.includes("#") && cOpacity != undefined) { - c += hexademicalOpacity[cOpacity] - } - return c - } else { - var c = cScale(d[cKey]) - if (c.includes("#") && cOpacity != undefined) { - c += hexademicalOpacity[cOpacity] - } else { - [r, g, b] = c.slice(3).split(',') - b = b.split(')').join('') - r = r.split('(').join('') - return 'rgb('+r+','+g+','+b+','+cOpacity+')' - } - return c - } - }) - .attr('stroke-width', strokeWidth) - .attr('stroke', function(d, i){ - if (cKey == undefined) { - var colorI = i % colors.length - var c = colors[colorI] - return c - } else { - return cScale(d[cKey]) - } - }) - - distributionContainer.attr('transform', function(d, i){ - var box = container.node().getBBox() - - var rx, ry, rr, tx, ty - - if (rotation == 'right') { - rr = 90 - rx = 0 - ry = 0 - tx = drawingSpace.y - ty = drawingSpace.x - } else if (rotation == 'left') { - rr = -90 - rx = 0 - ry = 0 - tx = 0 - ty = drawingSpace.x - } else if (rotation == 'down') { - rr = 180 - rx = drawingSpace.x / 2 - ry = drawingSpace.y / 2 - tx = - drawingSpace.x - ty = 0 - } else { - rr = 0 - rx = drawingSpace.x / 2 - ry = drawingSpace.y / 2 - tx = 0 - ty = 0 - } - - - var scale = 'scale(-1, 1)' - var rot = 'rotate('+rr+','+rx+','+ry+')' - var trans = 'translate('+tx+','+ty+')' - - if (rotation == 'down') { - return trans + rot + scale - } - if (rotation == 'right') { - return trans + rot + scale - } - return trans+rot - // return Math.abs(rotation / 180) >= 1 ? - // 'rotate('+rotation+',' + box.width / 2 + ',' + box.height + ')' - // : - // 'rotate('+rotation+',' + box.width + ',' + box.height / 2 + ')translate(0,'+-box.width+') ' - }) - - - - } - - - function setupSettersAndGetters() { - var closuredVariables = [ - 'container', - 'drawingSpace', - 'data', - 'colors', - 'vKey', 'cKey', - 'vScale', 'nScale', - 'vValues', 'nValues', - 'values', - 'cScale', - 'cOpacity', - 'strokeWidth', - 'xValues','yValues','cValues', - 'dataExtent', 'fatal', - 'verbose', 'animationTime', - 'svg', 'histogram', - 'rotation', - 'rangePad' - ] - - - for (var i = 0; i < closuredVariables.length; i++) { - var currentClosureVariable = closuredVariables[i] - var toEvaluateString = " \ - plot." + currentClosureVariable + " = function(value){ \ - if(!arguments.length) return " + currentClosureVariable + "; \ - " + currentClosureVariable + " = value; \ - return plot; \ - }; \ - " - eval(toEvaluateString) - } - } - - - setupSettersAndGetters() - return plot; -} diff --git a/needs-to-be-bundled/graph.js b/needs-to-be-bundled/graph.js deleted file mode 100644 index 4bbe410..0000000 --- a/needs-to-be-bundled/graph.js +++ /dev/null @@ -1,806 +0,0 @@ -function plot() { - var id, // svg ID - to='html', // scale svg to - x, // svg width either in pixels or percent - y, // svg height either in pixels or percetn - plot // plot type - - // internal variables - var svg - - - function configureSVG() { - if (to == undefined) { - svg.style("width", x + 'px') - svg.style("height", y + 'px') - return - } - - var el = d3.select(to).node() // element to scale to - var rect = el.getBoundingClientRect() // boundaries of the element - var css = getComputedStyle(el) // rendered css styles - - - var w = rect.width - parseFloat(css.paddingLeft) - parseFloat(css.paddingRight) // remove padding - var h = rect.height - parseFloat(css.paddingTop) - parseFloat(css.paddingBottom) - - var width, height - - if (String(x).includes('%')) { width = w * parseFloat(x) / 100 } - else if (x <= 1) { width = w * x} - else { width = x} - - if (String(y).includes('%')) { height = h * parseFloat(y) / 100 } - else if (x <= 1) { height = h * y} - else { height = y} - - - svg.style("width", width + 'px') - svg.style("height", height + 'px') - return - } - - function make() { - svg = d3.select('[id='+id+']') - configureSVG() - - plot.width(parseFloat(svg.style('width'))) - plot.height(parseFloat(svg.style('height'))) - - plot() - } - - function setupSettersAndGetters() { - var closuredVariables = [ - 'id', - 'to', - 'x', - 'y', - 'plot' - ] - - for (var i = 0; i < closuredVariables.length; i++) { - var currentClosureVariable = closuredVariables[i] - var toEvaluateString = " \ - make." + currentClosureVariable + " = function(value){ \ - if(!arguments.length) return " + currentClosureVariable + "; \ - " + currentClosureVariable + " = value; \ - return make; \ - }; \ - " - eval(toEvaluateString) - } - } - setupSettersAndGetters() - return make; - -} - - -function sugiyama() { - /* GRAPH DATA VARIABLES*/ - var vertexSet, - edgeSet, - dummySet, - realSet, - layering, - vertexKeys, - edgeKeys, - vertexMeta, - edgeMeta, - pathwayMeta, - pubmedMeta, - datasourceMeta - - - - /* GRAPH DRAWING VARIABLES*/ - var container, - vertexContainer, - vertexGroup, - edgeContainer, - edgeGroup, - width, - height, - - layout, - - xScale = d3.scaleLinear(), - yScale = d3.scaleLinear(), - - rangePad = 2, - - xCoordinates, - yCoordinates, - - dataExtent, - - zoom = d3.zoom() - .scaleExtent([0.25, 1.75]) - .on("zoom", containerZoom), - panRatio = 0.1 - - function containerZoom(){ - container.attr('transform', d3.event.transform) - } - - function containerZoomReset(){ - svg.call(zoom.transform, d3.zoomIdentity) - } - - /* GRAPH STYLE VARIABLES */ - var vertexRadius = 15, - vertexFill = '#333333', // vertex color - vertexStroke = '#333333', // vertex stroke - vertexStrokeWidth = 1, // vertex stroke - - textSize=7.5, // font size for vertex labels - textWeight = "900" // font weight for vertex labels - textStroke = '#333333', // stroke of vertex font (should match vertex color) - textStrokeWidth = 0.1, // stroke size of vertex font - textFill = 'white', // color of vertex font - - edgeLine = d3.line() - .x(function(d) { return d.x; }) - .y(function(d) { return d.y; }) - .curve(d3.curveMonotoneX), - // .curve(d3.curveBasis), - // .curve(d3.curveLinear), - - edgeStroke = "#333333", // color of edges - edgeStrokeMin = 2, - edgeStrokeMax = 10, - edgeStrokeScale = d3.scaleLinear().range([edgeStrokeMin, edgeStrokeMax]), - edgeStrokeWidth = edgeWidth, - - hoverOpacityDecrease = 0.2, - vertexIncreasedExpression = 'red', - vertexDecreasedExpression = 'blue' - - - function edgeWidth(edgeKey) { - var e = edgeSet[edgeKey] - if (edgeSet[edgeKey].meta) { - return edgeSet[edgeKey].meta.pubmed_ids.length - } else { - return edgeStrokeMin - } - } - - function vertexFillColor(vertexKey) { - var v = vertexSet[vertexKey] - - if (v.class.includes('dummy')) { return vertexFill } - - - if (v.expression) { - return (v.expression > 0) ? vertexIncreasedExpression : vertexDecreasedExpression - } else { - return vertexFill - } - } - - - function predraw() { - vertexKeys = d3.keys( vertexSet ) - edgeKeys = d3.keys( edgeSet ) - - xCoordinates = vertexKeys.map( function(d, i) { return vertexSet[d].coordinates.x }) - yCoordinates = vertexKeys.map( function(d, i) { return vertexSet[d].coordinates.y }) - - numPubs = edgeKeys.map( function (d, i) {if ( edgeSet[d].meta == undefined) {return edgeStrokeMin } return edgeSet[d].meta.pubmed_ids.length } ) - - dataExtent = { - x: { - min: d3.min( xCoordinates ), - max: d3.max( xCoordinates ) - }, - y: { - min: d3.min( yCoordinates ), - max: d3.max( yCoordinates ) - }, - pubs: { - min: d3.min( numPubs ), - max: d3.max( numPubs ) - } - - } - - - xScale.domain([ dataExtent.x.min - rangePad, dataExtent.x.max + rangePad ]) - .range([ 0 + vertexRadius, width - vertexRadius ]) - - yScale.domain([ dataExtent.y.min - rangePad, dataExtent.y.max + rangePad ]) - .range([ 0 + vertexRadius, height - vertexRadius ]) - - edgeStrokeScale.domain([ dataExtent.pubs.min, dataExtent.pubs.max ]) - - edgeLine.x(function(d) { return xScale(d.x); }) - .y(function(d) { return yScale(d.y); }) - - zoom.translateExtent([[width * panRatio * -1, height * panRatio * -1], [width * (1+panRatio), height * (1+panRatio)]]) - d3.select(container.thisSVG()).call(zoom) - - vertexContainer = container.select(".vertex-container") - vertexGroup = container.selectAll(".vertex-group") - - edgeContainer = container.select(".edge-container") - edgeGroup = container.selectAll(".edge-group") - - realSet = new Set([].concat.apply([],d3.keys(edgeSet).map(function(e,i){return [edgeSet[e].source, edgeSet[e].target]}))) - dummySet = new Set( d3.keys(vertexSet).map(function(v,i) {if (!realSet.has(v)) {return v}})) - } - - - function _vertex_setup() { - - if (vertexContainer.empty()) { - vertexContainer = container.append("g").attr("class", "vertex-container") - } else { - vertexContainer = container.select('.vertex-container') - } - - var neededVertices = vertexKeys.length - var currentVertices = vertexGroup.size() - if (currentVertices < neededVertices) { - vertexGroup = vertexContainer.selectAll(".vertex-group") - .data(vertexKeys).enter() - .append("g").attr("class", "vertex-group") - .attr("id", function(d, i){return d}) - - vertexGroup.each(function(d, i){ - var thisV = d3.select(this) - if (!dummySet.has(d)) { - thisV.append("circle").attr('class', 'vertex-shape') - thisV.append("text").attr('class', 'vertex-label') - } else { - thisV.classed("dummy", true) - thisV.append("rect").attr('class', 'vertex-shape') - } - }) - } else if (currentVertices > neededVertices) { - vertexGroup = vertexContainer.selectAll(".vertex-group") - .data(vertexKeys).exit().remove() - } else { - vertexGroup = vertexContainer.selectAll(".vertex-group").data(vertexKeys) - vertexGroup.each(function(d, i){ - var thisV = d3.select(this) - thisV.attr("id", function(d, i){return d}) - // console.log(d, dummySet.has(d)) - if (!dummySet.has(d)) { // real vertex - var circ, rect, text - circ = thisV.select('circle') - rect = thisV.select('rect') - text = thisV.select('text') - - if ( !rect.empty() ) { rect.remove() } - if ( circ.empty() ) { thisV.append('circle').attr('class', 'vertex-shape') } - text.attr('class', 'vertex-label') - - thisV.classed("dummy", false) - } else { // dummy vertex - var circ, rect, text - circ = thisV.select('circle') - rect = thisV.select('rect') - text = thisV.select('text') - - if ( rect.empty() ) { thisV.append('rect').attr('class', 'vertex-shape') } - if ( !circ.empty() ) { circ.remove() } - - thisV.classed("dummy", true) - } - }) - } - - vertexGroup = vertexContainer.selectAll(".vertex-group").data(vertexKeys) - - - - vertexGroup.on("mouseover", vertexMouseover) - vertexGroup.on("mouseout", vertexMouseout) - vertexGroup.call(d3.drag() - .on("start", vertexDragStart) - .on("drag", vertexDrag) - .on("end", vertexDragStop)); - - } - - - - function _vertex_style() { - - vertexGroup.attr("transform", function(d, i) { - var x = xScale(vertexSet[d].coordinates.x) - var y = yScale(vertexSet[d].coordinates.y) - var t = "translate("+x+","+y+")" - return t - }).each( function(d, i) { - var thisV = d3.select(this) - if (!dummySet.has(d)) { - thisV.select("text") - .text(d) - .attr("text-anchor", "middle") - .style("font-size",textSize) - .style("stroke", textStroke) - .style("stroke-width", textStrokeWidth) - .style("fill", textFill) - .style("font-weight", textWeight) - .style("cursor", "default") - .style("-webkit-user-select", "none") - .style("-moz-user-select", "none") - .style("-ms-user-select", "none") - .style("user-select", "none") - .attr("transform", function(d, i) { - var x = 0 - var y = textSize / 4 + textSize / 8 - var t = "translate("+x+","+y+")" - return t - }) - - thisV.select("circle") - .attr("stroke", vertexFillColor(d)) - .attr("fill", vertexFillColor(d)) - .attr("r", vertexRadius) - - } else { - thisV.select("rect") - .attr("stroke", vertexFillColor(d)) - .attr("fill", vertexFillColor(d)) - .attr("width", vertexRadius * 2) - .attr("height", vertexRadius * 2) - .attr("transform", function(d, i) { - var x = -vertexRadius - var y = -vertexRadius - var t = "translate("+x+","+y+")" - return t - }) - .style("opacity", 0) - } - }) - } - - function drawVertices() { - _vertex_setup() - _vertex_style() - } - - - function _edge_setup() { - - if (edgeContainer.empty()) { edgeContainer = container.append("g").attr("class", "edge-container") } - else { edgeContainer = container.select("g.edge-container")} - - var neededEdges = edgeKeys.length - var currentEdges = edgeContainer.size() - - if (currentEdges < neededEdges) { - edgeGroup = edgeContainer.selectAll(".edge-group") - .data(edgeKeys).enter() - .append("g").attr("class", "edge-group") - .attr("id", function(d, i){return d}) - - edgeGroup.each(function(d, i){ - thisE = d3.select(this) - thisE.append("path") - }) - } else if (neededEdges < currentEdges) { - edgeGroup = edgeContainer.selectAll(".edge-group") - .data(edgeKeys) - .exit().remove() - } else { - edgeGroup = edgeContainer.selectAll(".edge-group") - .data(edgeKeys) - } - - edgeGroup = edgeContainer.selectAll(".edge-group").data(edgeKeys) - - } - - function _edge_style() { - edgeGroup.each(function(d, i) { - var thisE = d3.select(this) - var coords = edgeSet[d].path.map(function(e, j) { return vertexSet[e].coordinates }) - thisE.select("path") - .attr("fill", "none") - .attr("stroke-width", edgeStrokeScale(edgeStrokeWidth(d))) - .attr("stroke", edgeStroke) - .attr("fill", "none") - .attr("d", edgeLine(coords)) - .on('mouseover', edgeMouseover) - .on('mouseout', edgeMouseout) - // .attr("stroke-dasharray", function(){ - // // console.log(d, d3.select(this)) - // return d3.select(this).node().getTotalLength() * 0.8 - // }) - - }) - edgeContainer.lower() - } - - function drawEdges() { - _edge_setup() - _edge_style() - } - - function draw() { - predraw() - drawVertices() - drawEdges() - } - - - function vertexMouseover(d, i) { - - var thisV = d3.select(this) - - if ( thisV.classed("dummy") ) { - thisV.select("rect").style("opacity","0.5") - return - } - - var adj = vertexSet[d].predecessors.concat(vertexSet[d].successors).concat([d]) - // console.log(d, adj) - - vertexKeys.map(function(dd, j){ - var cV = d3.select("[id='"+dd+"']") - if (adj.indexOf(dd) == -1) { - cV.style("opacity", hoverOpacityDecrease) - } - }) - - edgeKeys.map(function(dd, j){ - var cE = d3.select("[id='"+dd+"']") - if (edgeSet[dd].source != d && edgeSet[dd].target != d) { - cE.style("opacity", hoverOpacityDecrease) - } - }) - - - var ttip = d3.select('#graph-tooltip') - if (ttip.empty()) { ttip = d3.select("html").append('div').attr('id', 'graph-tooltip') } - var [div, title, body, close, txt, move] = setupTooltipCard( ttip ) - - if (!body.selectAll('*').empty()) {body.selectAll('*').remove()} - body.html('GeneCards: ') - - var bodyA = body.select('a') - if (bodyA.empty()) {bodyA = body.append('a'); bodyA.append('strong')} - bodyA.attr('href', 'http://www.genecards.org/cgi-bin/carddisp.pl?gene='+d) - .html(d).attr('class', 'gene-card').attr("target", '_blank') - - txt.html(d) - } - - function vertexMouseout(d, i) { - - var thisV = d3.select(this) - - d3.keys(vertexSet).map(function(dd, j){ - var cV = d3.select("[id='"+dd+"']") - cV.style("opacity", 1) - }) - - if ( thisV.classed("dummy") ) { - thisV.select("rect").style("opacity","0") - } - - d3.keys(edgeSet).map(function(dd, j){ - var cE = d3.select("[id='"+dd+"']") - cE.style("opacity", 1) - }) - - } - function vertexDragStart(){} - function vertexDrag(d, i){ - var curV = d3.select(this) - var m = d3.mouse(container.node()) - - vertexSet[d].coordinates.x = xScale.invert(m[0]) - vertexSet[d].coordinates.y = yScale.invert(m[1]) - - - curV.raise() - .attr("transform",function(d, i) { - x = m[0] - y = m[1] - trans = "translate("+x+","+y+")" - return trans - }) - - drawEdges() - } - function vertexDragStop(){} - - function edgeMouseout(d, i) { - var ttip = d3.select('#graph-tooltip') - // if (!ttip.empty()) { ttip.remove() } - } - - - function makeCloseButton(closeSelection, containerSelection) { - if (closeSelection.empty()) {closeSelection = containerSelection.append('button') - .attr('type', 'button') - .attr('class', 'close align-middle tt-close') - .attr('data-dismiss', 'alert') - .on('click', function(){d3.select("#graph-tooltip").transition().duration(300).style('opacity', 0).remove()}) - .append('i').attr('class', "fa fa-times fa-lg") - .style('color', '#000') - } - } - function makeMoveButton(moveSelection, containerSelection, toMoveSelection) { - if (moveSelection.empty()) { - moveSelection = containerSelection.append('button') - .attr('class', 'move close') - .call(d3.drag().on('drag', function(){ - var m = d3.mouse(moveSelection.node()) - toMoveSelection.style('left', (parseFloat(toMoveSelection.style('left'))+m[0])+'px') - .style('top', (parseFloat(toMoveSelection.style('top'))+m[1])+'px') - })) - .style('float','right') - .style('padding','none') - .append('i').attr('class','fa fa-arrows-alt fa-sm') - .style('color', '#000') - .style('margin-right', '3px') - } - } - function movePercentInsideHtml(div, mouse, percent) { - var html = d3.select('html').node().getBoundingClientRect() - var left = Math.min( mouse[0], html.width - (html.width * percent) ) - div.transition().duration(200).style('left', left+'px').style('top', mouse[1]+'px') - } - - function setupTooltipCard( tooltip ) { - var div = tooltip.select('div.card') - var m = d3.mouse.absolute() - - if (div.empty()) { div = tooltip.append('div').attr('class', 'card col-3 px-0').style('position', 'absolute').style('left', m[0]+'px').style('top', m[1]+'px') } - movePercentInsideHtml(div, m, 0.25) - - var title, body, footer, datasources, interactions, pubmedids, pathways, close, txt, move - title = div.select('div.card-header') - body = div.select('div.card-body') - close = title.select('button.tt-close') - txt = title.select('h6') - move = title.select('button.move') - - if (title.empty()) {title = div.append('div').attr('class','card-header card-title')} - if (txt.empty()) {txt = title.append('h6').style('display', 'inline-block')} - - makeCloseButton(close, title) - makeMoveButton(move, title, div) - - if (body.empty()) {body = div.append('div').attr('class','card-body')} - return [div, title, body, close, txt, move] - } - - function edgeMouseover(d, i) { - var thisE = d3.select(this) - if (edgeMeta == undefined) { return } - - var ttip = d3.select('#graph-tooltip') - if (ttip.empty()) { ttip = d3.select("html").append('div').attr('id', 'graph-tooltip') } - - var [div, title, body, close, txt, move] = setupTooltipCard( ttip ) - - var bodyA = body.select('a.gene-card') - if (!bodyA.empty()) {bodyA.remove()} - - var bodyUL = body.select('ul.list-group') - if (bodyUL.empty()) {bodyUL = body.append('ul').attr('class', 'list-group well')} - - var metakeys = [] - var raw_metakeys = ['datasources', "interactions", 'pathways', 'pubmed_ids'] - raw_metakeys.map(function(k, i){ - if((edgeMeta[d][k].length > 0) && (d3.keys(edgeMeta[d][k]).length > 0)) {metakeys.push(k)} - }) - - var bodyUlLis = setupTooltipCardBodyUl( bodyUL, metakeys ) - fillTooltipCardBodyUlList ( d, bodyUlLis, edgeMeta ) - - txt.html(d.split('_')[0] + '' + d.split('_')[1]) - } - - function setupTooltipCardBodyUl( bodyUL, metakeys ) { - var bodyUlLis = bodyUL.selectAll('li.outer-item') - if (metakeys.length > bodyUlLis.size()) { - bodyUlLis = bodyUL.selectAll('li.outer-item') - .data(metakeys).enter() - .append('li') - .attr('class', 'list-group-item list-group-item-action outer-item toolbar-list-group-item') - .attr('href', function(d){return'#'+'graph-tooltip-'+d}).attr('data-toggle', 'collapse') - - bodyUlLis.append('i').attr('class', 'fa fa-caret-right') - bodyUlLis.append('h6').style('display', 'inline') - - bodyUlLis.on('click', function() { - var i = d3.select(this).select('i') - if (i.classed('fa-caret-right')) {i.classed('fa-caret-down', true).classed('fa-caret-right', false)} - else {i.classed('fa-caret-down', false).classed('fa-caret-right', true)} - }) - - } else if (metakeys.length < bodyUlLis.size()) { - bodyUlLis = bodyUL.selectAll('li.outer-item').data(metakeys).exit().remove() - } else { - bodyUlLis = bodyUL.selectAll('li.outer-item').data(metakeys) - } - bodyUlLis = bodyUL.selectAll('li.outer-item').data(metakeys) - - return bodyUlLis - } - - function fillTooltipCardBodyUlList ( edgeKey, bodyUlLis, edgeMeta ) { - bodyUlLis.selectAll('h6').html(function(d, i){return d}) - bodyUlLis.each(function (dd, ii) { - var li = d3.select(this) - var liUl = li.select('ul') - if (liUl.empty()) { - liUl = li.append('ul') - .attr('class', 'list-group collapse indent-1') - .attr('id',function(d){return 'graph-tooltip-'+d}) - .style('max-height', '200px') - .style('overflow-y', 'scroll') - } - var dataToUse = edgeMeta[edgeKey][dd] - dataToUse = (dd == 'pathways') ? d3.keys(dataToUse) : dataToUse - var liUlLis = liUl.selectAll('li') - - if (liUlLis.size() < dataToUse.length) { liUlLis = liUl.selectAll('li').data(dataToUse).enter().append('li').attr('class', 'list-group-item list-group-item-action toolbar-list-group-item') } - else if (liUlLis.size() > dataToUse.length) { liUlLis = liUl.selectAll('li').data(dataToUse).exit().remove() } - else { liUlLis = liUl.selectAll('li').data(dataToUse) } - - if (dd == 'datasources') { _makeSubListFromMeta(liUlLis, datasourceMeta) } - else if (dd == 'pathways') { _makeSubListFromMeta(liUlLis, pathwayMeta) } - else if (dd == 'pubmed_ids') { _makeSubListFromMeta(liUlLis, pubmedMeta, 'uid', 'http://ncbi.nlm.nih.gov/pubmed/', 'title') } - else {_makeSubList(liUlLis)} - }) - } - - function _makeSubList(liSelection) { - liSelection.each(function(d, i) { - var li = d3.select(this) - li.html(d) - }) - } - - function _makeSubListFromMeta(liSelection, meta, urlKey, urlPrefix, urlTextKey) { - liSelection.each(function(d, i) { - var li = d3.select(this) - var a = li.select('a') - var urlText - if (meta != undefined && meta[d] != undefined) { - if (urlKey == undefined) {urlKey = 'url'} - if (urlPrefix == undefined) {urlPrefix == ''} - if (urlTextKey == undefined) {urlText = d} else {urlText = meta[d][urlTextKey]} - if (a.empty()) {li.html(''); a = li.append('a')} - a.attr('href', urlPrefix + meta[d][urlKey]).attr('target', '_blank').html(urlText) - } else { - if (!a.empty()) {a.remove()} - li.html(d) - } - }) - } - - - - - function setupSettersAndGetters() { - var closuredVariables = [ - 'vertexSet', - 'edgeSet', - 'layering', - /* GRAPH DRAWING VARIABLES*/ - 'container', - 'width', - 'height', - 'layout', - 'xScale', - 'yScale', - 'rangePad', - 'xCoordinates', - 'yCoordinates', - 'dataExtent', - /* GRAPH STYLE VARIABLES */ - 'vertexRadius', - 'vertexContainer', - 'edgeContainer', - - 'vertexGroup', - 'edgeGroup', - 'vertexMeta', - 'edgeMeta', - 'pathwayMeta', - 'pubmedMeta', - 'datasourceMeta' - ] - - for (var i = 0; i < closuredVariables.length; i++) { - var currentClosureVariable = closuredVariables[i] - var toEvaluateString = " \ - draw." + currentClosureVariable + " = function(value){ \ - if(!arguments.length) return " + currentClosureVariable + "; \ - " + currentClosureVariable + " = value; \ - return draw; \ - }; \ - " - eval(toEvaluateString) - } - } - setupSettersAndGetters() - return draw; -} - - -function marker() { - var id, - markerWidth=10, - markerHeight=10, - refX=0, - refY=3, - orient="auto", - markerUnits="strokeWidth", - viewBox="0 0 20 20", - path="M0,0 L0,6 L9,3 z", - fill="#333333", - type="triangle" - - - function make() { - var defs = d3.select(svg).select("defs") - if (defs.empty()) { - defs = d3.select(svg).append("defs") - } - defs.append("marker") - .attr("id", id) - .attr("markerWidth", markerWidth) - .attr("markerHeight", markerHeight) - .attr("refX", refX) - .attr("refY", refY) - .attr("orient", orient) - .attr("markerUnits", markerUnits) - .attr("viewBox", viewBox) - .append("path") - .attr("d", path) - } - - make.triangle = function (x, y, yOffset) { - refX = yOffset + y - refY = x / 2 - path = "M0,0 L0,"+x+" L"+y+","+(x/2)+" z" - } - - function setupSettersAndGetters() { - var closuredVariables = [ - 'id', 'markerWidth', 'markerHeight', 'refX', 'refY', - 'orient', 'markerUnits', 'viewBox', 'path', "svg", "fill" - ] - - - - for (var i = 0; i < closuredVariables.length; i++) { - var currentClosureVariable = closuredVariables[i] - var toEvaluateString = " \ - make." + currentClosureVariable + " = function(value){ \ - if(!arguments.length) return " + currentClosureVariable + "; \ - " + currentClosureVariable + " = value; \ - return make; \ - }; \ - " - eval(toEvaluateString) - } - } - setupSettersAndGetters() - return make; -} - - - -// marker().id("arrow-head") -// .markerWidth("10") -// .markerHeight("10") -// .refX("0") -// .refY("3") -// .orient("auto") -// .markerUnits("strokeWidth") -// .viewBox("0 0 20 20") -// .path("M0,0 L0,6 L9,3 z") -// .fill("#FFFFFF") diff --git a/needs-to-be-bundled/scatter.js b/needs-to-be-bundled/scatter.js deleted file mode 100644 index 3c97ada..0000000 --- a/needs-to-be-bundled/scatter.js +++ /dev/null @@ -1,265 +0,0 @@ -function scatter() { - // variables that need to be set externally - var plotContainer, - drawingSpace, - colors = ["#2c7bb6", "#00a6ca", "#00ccbc", "#90eb9d", "#ffff8c", "#f9d057", "#f29e2e", "#e76818", "#d7191c"], - tooltipKeys, - rKey, // r key - r=2, - stroke, - data, - xScale = d3.scaleLinear(), - yScale = d3.scaleLinear(), - rScale = d3.scaleLinear().range([ 2, 10]), - xKey, yKey, // x and y keys - points, - categorical = true, - cKey, // color key - cOpacity = 0.2, - strokeWidth = 2, - cScale = d3.scaleSequential(interpolateColors.apply(this, colors)) - - - - // variables set internally - var pointsNeeded, currentNumberOfPoints, - xValues, yValues, rValues, cValues, - dataExtent, fatal, - verbose = true, - animationTime = 1000, - svg, - rScaleMin = 0.002, - rScaleMax = 0.02, - xRangePad = 2, - yRangePad = 2 - hexademicalOpacity = { - 1: "FF", - 0.9: 'E6', - 0.8: 'CC', - 0.7: "B3", - 0.6: "99", - 0.5: "80", - 0.4: "66", - 0.3: "4D", - 0.2: "33", - 0.1: "1A", - 0.0: "00" - } - - - - - function plot(){ - fatal = warn(plotContainer, 'plotContainer', undefined, true, verbose) - if ( fatal ) {return} - fatal = warn(data, 'data', undefined, true, verbose) - if ( fatal ) {return} - fatal = warn(xKey, 'xKey', undefined, true, verbose) - if ( fatal ) {return} - fatal = warn(yKey, 'yKey', undefined, true, verbose) - if ( fatal ) {return} - fatal = warn(drawingSpace, 'drawingSpace', undefined, true, verbose) - if ( fatal ) {return} - - - svg = d3.select(plotContainer.thisSVG()) - - pointsNeeded = data.length - currentNumberOfPoints = plotContainer.selectAll('.point-container').size() - - - xValues = data.map( function(d, i) { return parseFloat( d[xKey] ) } ) - yValues = data.map( function(d, i) { return parseFloat( d[yKey] ) } ) - - - - dataExtent = { - x: { - min: d3.min( xValues ), - max: d3.max( xValues ) - }, - y: { - min: d3.min( yValues ), - max: d3.max( yValues ) - } - } - - xScale.domain([ dataExtent.x.min - xRangePad, dataExtent.x.max + xRangePad]) - .range([ 0, drawingSpace.x ]) - - yScale.domain([ dataExtent.y.min - yRangePad, dataExtent.y.max + yRangePad]) - .range([ drawingSpace.y, 0 ]) - - - // set r scale if rKey is defined - if (rKey != undefined) { - rValues = data.map( function(d, i) { return parseFloat( d[rKey] ) } ) - rScale.domain([ d3.min(rValues), d3.max(rValues) ]) - .range([ - parseFloat(svg.style('width')) * rScaleMin, - parseFloat(svg.style('width')) * rScaleMax] - ) - } - - if (cKey != undefined) { - cValues = data.map( function(d, i) { return parseFloat( d[cKey] ) } ) - cScale.domain([ d3.min(cValues), d3.max(cValues) ]) - } - - - // not enough pionts add more - if (pointsNeeded > currentNumberOfPoints) { - plotContainer.selectAll('.point-container') - .data(data) - .enter() - .append('g') - .attr('class', 'point-container') - .attr('transform', function(d, i) { - x = 0 - y = parseFloat(plotContainer.thisSVG().style.height) - trans = 'translate('+x+','+y+')' - return trans - }) // for entering animation - .append('circle') - .attr('class', 'point') - } else if (pointsNeeded == currentNumberOfPoints) { - plotContainer.selectAll('.point-container').data(data) - } else { // too many points, remove - plotContainer.selectAll('.point-container').data(data) - .exit() - .transition() - .duration(function(d, i) { - return (animationTime * 0.1) + (animationTime * .9) / data.length * i - }) - .attr('transform', function(d, i) { - x = 0 - y = parseFloat(svg.style('height')) - trans = 'translate('+x+','+y+')' - return trans - }) // for exiting animation - .remove() - } - - // rebind dat for safety - points = plotContainer.selectAll('.point-container').data(data) - - points.transition().duration(function(d, i){ - return (animationTime * 0.1) + (animationTime * .9) / data.length * i - }) // animate movement - .attr('transform', function(d, i) { - x = xScale(d[xKey]) - y = yScale(d[yKey]) - trans = "translate("+(x)+","+(y)+")" - return trans - }) // move points to correct location - .select('circle') - .attr('r', function(d, i){ - if (rKey != undefined) { return rScale(d[rKey]) } - else { return r } - }) // set size to r - .attr('fill', function(d, i){ - if (cKey == undefined) { - var colorI = i % colors.length - var c = colors[colorI] - if (c.includes("#") && cOpacity != undefined) { - c += hexademicalOpacity[cOpacity] - } - return c - } else { - var c = cScale(d[cKey]) - if (c.includes("#") && cOpacity != undefined) { - c += hexademicalOpacity[cOpacity] - } else { - [r, g, b] = c.slice(3).split(',') - b = b.split(')').join('') - r = r.split('(').join('') - return 'rgba('+r+','+g+','+b+','+cOpacity+')' - } - return c - } - }) - .attr('stroke-width', strokeWidth) - .attr('stroke', function(d, i){ - if (cKey == undefined) { - var colorI = i % colors.length - var c = colors[colorI] - return c - } else { - return cScale(d[cKey]) - } - }) - - - - points.on('mouseout', mouseoutPoint) - .on('mouseover', mouseoverPoint) - - function mouseoutPoint(d, i){ - var pnt = d3.select(this).select('circle') - d3.selectAll('div.tooltip').style('display', 'none') - pnt.transition().duration(animationTime * 0.5) - .attr('r', function(d, i){ - if (rKey != undefined) { return rScale(d[rKey]) } - else { return r } - }) - } - - function mouseoverPoint(d, i){ - var pnt = d3.select(this).select('circle') - pnt.transition().duration(animationTime * 0.5) - .attr('r', function(d, i){ - return rScale.range()[1] * 1.5 - }) - - ttip = tooltip().tooltipKeys(tooltipKeys).data(d) - ttip() - } - - } - - - function setupSettersAndGetters() { - var closuredVariables = [ - 'plotContainer', - 'drawingSpace', - 'data', - 'colors', - 'tooltipKeys', - 'rKey','xKey', 'yKey', 'cKey', - 'r', - 'xScale', - 'yScale', - 'cScale', - 'rScale', - 'xRangePad', - 'yRangePad', - 'points', - 'categorical', - 'cOpacity', - 'strokeWidth', - 'pointsNeeded', - 'currentNumberOfPoints', - 'xValues','yValues','rValues','cValues', - 'dataExtent', 'fatal', - 'verbose', 'animationTime', - 'svg', 'rScaleMax', 'rScaleMin' - ] - - - for (var i = 0; i < closuredVariables.length; i++) { - var currentClosureVariable = closuredVariables[i] - var toEvaluateString = " \ - plot." + currentClosureVariable + " = function(value){ \ - if(!arguments.length) return " + currentClosureVariable + "; \ - " + currentClosureVariable + " = value; \ - return plot; \ - }; \ - " - eval(toEvaluateString) - } - } - - - setupSettersAndGetters() - return plot; -} diff --git a/needs-to-be-bundled/tooltip.js b/needs-to-be-bundled/tooltip.js deleted file mode 100644 index c4ba690..0000000 --- a/needs-to-be-bundled/tooltip.js +++ /dev/null @@ -1,79 +0,0 @@ -function tooltip() { - var - ttip, - tooltipKeys, - tooltipContent, - divs, data - - - function make() { - ttip = d3.select('body').select('div.tooltip') - .style('display', 'flex') - - if (ttip.empty()) { - ttip = d3.select('body').append('div').attr('class','tooltip') - .style('display', 'flex').attr('id', 'tooltip') - } - - tooltipContent = ttip.select('div.tooltip-vertical-flex') - if (tooltipContent.empty()) { - tooltipContent = ttip.append('div').attr('class','tooltip-vertical-flex') - } - - divs = tooltipContent.selectAll('.tooltip-element-flex') - if (divs.empty() || divs.size() < tooltipKeys.length) { - divs = tooltipContent.selectAll('.tooltip-element-flex') - .data(tooltipKeys) - .enter() - .append('div').attr('class', 'tooltip-element-flex') - - divs.append('div').attr('class', 'tooltip-element-key') - divs.append('div').attr('class', 'tooltip-element-value') - } else { - tooltipContent.selectAll('.tooltip-element-flex') - .data(tooltipKeys) - .exit() - .remove() - } - divs = tooltipContent.selectAll('.tooltip-element-flex').data(tooltipKeys) - - divs.each(function(curKey){ - var k = d3.select(this).select('.tooltip-element-key') - var v = d3.select(this).select('.tooltip-element-value') - - k.html(curKey) - v.html(data[curKey]) - }) - - var mouse = d3.mouse.smart("tooltip") - ttip.style('left', mouse[0]+'px') - .style('top', mouse[1]+'px') - - } - - function setupSettersAndGetters() { - var closuredVariables = [ - 'ttipConatiner', - 'ttip', - 'tooltipKeys', - 'tooltipContent', - 'divs', 'data' - ] - - - for (var i = 0; i < closuredVariables.length; i++) { - var currentClosureVariable = closuredVariables[i] - var toEvaluateString = " \ - make." + currentClosureVariable + " = function(value){ \ - if(!arguments.length) return " + currentClosureVariable + "; \ - " + currentClosureVariable + " = value; \ - return make; \ - }; \ - " - eval(toEvaluateString) - } - } - - setupSettersAndGetters() - return make -} diff --git a/rollup.config.js b/rollup.config.js index f638afd..186a4da 100644 --- a/rollup.config.js +++ b/rollup.config.js @@ -1,8 +1,8 @@ -import babel from 'rollup-plugin-babel'; +// import babel from 'rollup-plugin-babel'; import uglify from 'rollup-plugin-uglify-es'; import minimist from 'minimist'; import commonjs from 'rollup-plugin-commonjs'; -import resolve from 'rollup-plugin-node-resolve'; +import nodeResolve from 'rollup-plugin-node-resolve'; import pkg from './package.json'; @@ -10,39 +10,25 @@ import pkg from './package.json'; const argv = minimist(process.argv.slice(2)); const config = { - input: 'src/scripts/main.js', + input: 'src/entry.js', output: { name: pkg.name, exports: 'named', extend: true, - // external: ['d3'], - // globals: { - // d3: 'd3' - // }, + external: ['d3'], + globals: { + d3: 'd3' + }, }, plugins: [ - // babel({ // exclude: 'node_modules/**', // externalHelpers: true, - // runtimeHelpers: true, - // plugins: [ - // [ - // 'wildcard', - // { - // exts: ['vue'], - // nostrip: true, - // }, - // ], - // '@babel/plugin-external-helpers', - // - // ], + // runtimeHelpers: true, // }), - resolve({ - jsnext: true, - main: true - + nodeResolve({ + mainFields:['jsnext', 'main'], }), commonjs({ // non-CommonJS modules will be ignored, but you can also diff --git a/src/entry.js b/src/entry.js new file mode 100644 index 0000000..2ac3867 --- /dev/null +++ b/src/entry.js @@ -0,0 +1,27 @@ +import utils from './modules/utils/index.js' +import axis from './modules/axis' +import groupingSpacer from './modules/grouping-spacer' +import tooltip from './modules/tooltip' +import colorFunction from './modules/color-function' +import charts from './modules/charts/index.js' +import legends from './modules/legends/index.js' +import aux from './modules/aux/index.js' + + + +let d3sm = { + aux, + axis, + charts, + colorFunction, + debugQ: false, + groupingSpacer, + legends, + tooltip, + utils, +} + +if (typeof window !== 'undefined') { + window.d3sm = d3sm; +} +export default d3sm diff --git a/src/modules/aux/index.js b/src/modules/aux/index.js new file mode 100644 index 0000000..c52b890 --- /dev/null +++ b/src/modules/aux/index.js @@ -0,0 +1,9 @@ +import lasso from './lasso' +import multiPlotZoom from './multi-plot-zoom' +import plotZoom from './plot-zoom' +let aux = { + lasso, plotZoom, multiPlotZoom +} + +export {lasso, plotZoom, multiPlotZoom} +export default aux diff --git a/src/scripts/modules/lasso.js b/src/modules/aux/lasso.js similarity index 87% rename from src/scripts/modules/lasso.js rename to src/modules/aux/lasso.js index 0ccf0e9..d3bbb24 100644 --- a/src/scripts/modules/lasso.js +++ b/src/modules/aux/lasso.js @@ -1,20 +1,18 @@ -import {hypenate, safeSelect, euclideanDistance} from './helpers'; -import {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer} from './utils'; -import {unique, hasQ, flatten, whichBin} from './array-functions'; -import {groupingSpacer} from './grouping-spacer'; -import {colorFunction as CF} from './color-function'; -import {tooltip as TTip} from './tooltip'; -import './d3-prototypes'; - -function getTranslation(selection){ - var transform = selection.attr('transform') - var [junk, xy] =transform.split('translate(') - var [x, y] = xy.split(',') - y, junk = y.split(')') - return [parseFloat(x), parseFloat(y)] -} - -export function lasso( selection ) { +import utils from '../utils/index.js'; +import groupingSpacer from '../grouping-spacer'; +import CF from '../color-function'; +import TTip from '../tooltip'; +import '../d3-prototypes'; + +// function getTranslation(selection){ +// var transform = selection.attr('transform') +// var [junk, xy] =transform.split('translate(') +// var [x, y] = xy.split(',') +// y, junk = y.split(')') +// return [parseFloat(x), parseFloat(y)] +// } + +export default function lasso( selection ) { var svg, // svg that is target of events objectContainer, // container which houses objects we are selecting (allows for transform to be applied to lasso) @@ -114,8 +112,8 @@ export function lasso( selection ) { function toggle(state) { // use optional param to set state, otherwise toggle state activeQ = (state!=undefined) ? state : !activeQ - chartOffset = getTranslation(chartContainer) - objectsOffset = getTranslation(objectContainer) + chartOffset = utils.sel.getTranslation(chartContainer) + objectsOffset = utils.sel.getTranslation(objectContainer) if (activeQ) { svg.node().addEventListener('mousedown', render, true) @@ -127,10 +125,10 @@ export function lasso( selection ) { } function draw() { - chartOffset = getTranslation(chartContainer) - objectsOffset = getTranslation(objectContainer) + chartOffset = utils.sel.getTranslation(chartContainer) + objectsOffset = utils.sel.getTranslation(objectContainer) - var container = safeSelect(objectContainer, 'g', 'lasso-container') + var container = utils.sel.safeSelect(objectContainer, 'g', 'lasso-container') var paths = container.selectAll('path[instance="'+instance+'"]') // update @@ -149,7 +147,7 @@ export function lasso( selection ) { } function remove() { - var container = safeSelect(objectContainer, 'g', 'lasso-container') + var container = utils.sel.safeSelect(objectContainer, 'g', 'lasso-container') var paths = container.selectAll('path[instance="'+instance+'"]').remove() container.remove() objectContainer.selectAll(objectClass).classed("in-lasso", false) @@ -160,7 +158,7 @@ export function lasso( selection ) { // nothing can interefer with drawing the lasso event.preventDefault(); event.stopPropagation(); - var container = safeSelect(objectContainer, 'g', 'lasso-container') + var container = utils.sel.safeSelect(objectContainer, 'g', 'lasso-container') /* each time the user presses down, while the state is active, the lasso @@ -173,8 +171,8 @@ export function lasso( selection ) { svg.node().removeEventListener('mousemove', drag) allPoints.push(currentPoints) // BUG: somehow this is pushing currentPoints n times where n is the nth lasso path for the current instance - // NOTE: allPoints = unique(allPoints) is a temporary and inefficient fix - allPoints = unique(allPoints) + // NOTE: allPoints = utils.arr.unique(allPoints) is a temporary and inefficient fix + allPoints = utils.arr.unique(allPoints) }) path = container.append('path').data([currentPoints]) @@ -182,7 +180,7 @@ export function lasso( selection ) { } function transitionDraw() { - var container = safeSelect(objectContainer, 'g', 'lasso-container') + var container = utils.sel.safeSelect(objectContainer, 'g', 'lasso-container') var paths = container.selectAll('path[instance="'+instance+'"]') // update @@ -203,7 +201,7 @@ export function lasso( selection ) { function applyPathAttributes(path) { path - .attr("class", hypenate(namespace, "lasso-path")) + .attr("class", utils.str.hypenate(namespace, "lasso-path")) .style('opacity', opacity) .attr('fill', color) .attr("d", line) @@ -222,8 +220,8 @@ export function lasso( selection ) { mouse down / mouse up. */ - if (eventCatcher != undefined) {eventCatcher.dispatch(hypenate(namespace,"drag"))} - // d3.dispatch(hypenate(namespace,"drag")) + if (eventCatcher != undefined) {eventCatcher.dispatch(utils.str.hypenate(namespace,"drag"))} + // d3.dispatch(utils.str.hypenate(namespace,"drag")) if (event.which != 1) {return} // ensures left mouse button set d3.event = event @@ -243,7 +241,7 @@ export function lasso( selection ) { if (xScale) {b[0] = xScale(b[0]); a[0] = xScale(a[0])} if (yScale) {b[1] = yScale(b[1]); a[1] = yScale(a[1])} - var dist = euclideanDistance(b, a) + var dist = utils.math.euclideanDistance(b, a) if (dist > tickDistance) { tick(pt) } } else { tick(pt) } @@ -379,10 +377,10 @@ export function lasso( selection ) { function keyFrames() { var style = - d3.select("html").select('style.'+hypenate(namespace,"lasso-dash")) + d3.select("html").select('style.'+utils.str.hypenate(namespace,"lasso-dash")) if (style.empty()) { d3.select("html").append('style') - .classed(hypenate(namespace,"lasso-dash"), true) + .classed(utils.str.hypenate(namespace,"lasso-dash"), true) .html("@keyframes lassoDash {to { stroke-dashoffset: 1000;}}") } diff --git a/src/scripts/modules/multi-plot-zoom.js b/src/modules/aux/multi-plot-zoom.js similarity index 88% rename from src/scripts/modules/multi-plot-zoom.js rename to src/modules/aux/multi-plot-zoom.js index 1565e4a..18db8b3 100644 --- a/src/scripts/modules/multi-plot-zoom.js +++ b/src/modules/aux/multi-plot-zoom.js @@ -1,5 +1,4 @@ -import {hypenate, safeSelect, getTranslation} from './helpers'; -import {log, warn, error, info} from './utils'; +import utils from '../utils/index.js' /******************************************************************************* ** ** @@ -17,7 +16,7 @@ import {log, warn, error, info} from './utils'; * @namespace plotZoom * @returns {function} zoom */ -export function multiPlotZoom( chart ) { +export default function multiPlotZoom( chart ) { var /** * The event on which to fire @@ -127,14 +126,14 @@ export function multiPlotZoom( chart ) { function setLocks() { - var chartObjSel = chartSel.select('.'+hypenate(chart.namespace(),'object-container')) - var chartObjTrans = getTranslation(chartObjSel.attr('transform')) + var chartObjSel = chartSel.select('.'+utils.str.hypenate(chart.namespace(),'object-container')) + var chartObjTrans = utils.sel.getTranslation(chartObjSel.attr('transform')) var cos = chartObjSel.attr('transform', 'translate(0,0)') xLock = chartSel.node().getBBox().width - chart.spaceX()// * .9 yLock = chartSel.node().getBBox().height - chart.spaceY()// * .9 cos.attr('transform', 'translate('+chartObjTrans[0]+','+chartObjTrans[1]+')') - log('plotZoom', 'setLocks', {xLock:xLock, yLock:yLock}) + utils.con.log('plotZoom', 'setLocks', {xLock:xLock, yLock:yLock}) } @@ -194,20 +193,20 @@ export function multiPlotZoom( chart ) { - var chartObjSel = chartSel.select('.'+hypenate(chart.namespace(),'object-container')) + var chartObjSel = chartSel.select('.'+utils.str.hypenate(chart.namespace(),'object-container')) var xComponentObjSel = xComponentsSel.map(function(d, i){ - return d.select('.'+hypenate(xComponents[i].namespace(),'object-container')) + return d.select('.'+utils.str.hypenate(xComponents[i].namespace(),'object-container')) }) var yComponentObjSel = yComponentsSel.map(function(d, i){ - return d.select('.'+hypenate(yComponents[i].namespace(),'object-container')) + return d.select('.'+utils.str.hypenate(yComponents[i].namespace(),'object-container')) }) - var chartObjTrans = getTranslation(chartObjSel.attr('transform')) + var chartObjTrans = utils.sel.getTranslation(chartObjSel.attr('transform')) var xComponentsObjTrans = xComponentObjSel.map(function(d, i){ - return getTranslation(d.attr('transform')) + return utils.sel.getTranslation(d.attr('transform')) }) var yComponentsObjTrans = yComponentObjSel.map(function(d, i){ - return getTranslation(d.attr('transform')) + return utils.sel.getTranslation(d.attr('transform')) }) var x = horizontalQ ? transform.applyX(chartObjTrans[0]) : 0 @@ -236,9 +235,9 @@ export function multiPlotZoom( chart ) { if (orient == 'horizontal') {horizontalQ = true; verticalQ = false;} if (orient == 'vertical') {verticalQ = true; horizontalQ = false;} - var chartObjSel = chartSel.select('.'+hypenate(chart.namespace(),'object-container')) - var xAxisObjSel = xAxisSel.select('.'+hypenate(xAxis.namespace(),'object-container')) - var yAxisObjSel = yAxisSel.select('.'+hypenate(yAxis.namespace(),'object-container')) + var chartObjSel = chartSel.select('.'+utils.str.hypenate(chart.namespace(),'object-container')) + var xAxisObjSel = xAxisSel.select('.'+utils.str.hypenate(xAxis.namespace(),'object-container')) + var yAxisObjSel = yAxisSel.select('.'+utils.str.hypenate(yAxis.namespace(),'object-container')) chartObjSel.attr('transform', 'translate('+0+','+0+')') xAxisObjSel.attr('transform', 'translate('+0+','+0+')') yAxisObjSel.attr('transform', 'translate('+0+','+0+')') diff --git a/src/scripts/modules/plot-zoom.js b/src/modules/aux/plot-zoom.js similarity index 87% rename from src/scripts/modules/plot-zoom.js rename to src/modules/aux/plot-zoom.js index 6cca5f6..6aef3f4 100644 --- a/src/scripts/modules/plot-zoom.js +++ b/src/modules/aux/plot-zoom.js @@ -1,5 +1,4 @@ -import {hypenate, safeSelect, getTranslation} from './helpers'; -import {log, warn, error, info} from './utils'; +import utils from '../utils/index.js'; /******************************************************************************* ** ** @@ -17,7 +16,7 @@ import {log, warn, error, info} from './utils'; * @namespace plotZoom * @returns {function} zoom */ -export function plotZoom( chart, xAxis, yAxis ) { +export default function plotZoom( chart, xAxis, yAxis ) { var /** * The event on which to fire @@ -122,13 +121,13 @@ export function plotZoom( chart, xAxis, yAxis ) { function setLocks() { - var chartObjSel = chartSel.select('.'+hypenate(chart.namespace(),'object-container')) - var chartObjTrans = getTranslation(chartObjSel.attr('transform')) + var chartObjSel = chartSel.select('.'+utils.str.hypenate(chart.namespace(),'object-container')) + var chartObjTrans = utils.sel.getTranslation(chartObjSel.attr('transform')) var cos = chartObjSel.attr('transform', 'translate(0,0)') xLock = chartSel.node().getBBox().width - chart.spaceX() * .9 yLock = chartSel.node().getBBox().height - chart.spaceY() * .9 cos.attr('transform', 'translate('+chartObjTrans[0]+','+chartObjTrans[1]+')') - log('plotZoom', 'setLocks', {xLock:xLock, yLock:yLock}) + utils.con.log('plotZoom', 'setLocks', {xLock:xLock, yLock:yLock}) } @@ -187,9 +186,9 @@ export function plotZoom( chart, xAxis, yAxis ) { - var chartObjSel = chartSel.select('.'+hypenate(chart.namespace(),'object-container')) - var xAxisObjSel = xAxisSel.select('.'+hypenate(xAxis.namespace(),'object-container')) - var yAxisObjSel = yAxisSel.select('.'+hypenate(yAxis.namespace(),'object-container')) + var chartObjSel = chartSel.select('.'+utils.str.hypenate(chart.namespace(),'object-container')) + var xAxisObjSel = xAxisSel.select('.'+utils.str.hypenate(xAxis.namespace(),'object-container')) + var yAxisObjSel = yAxisSel.select('.'+utils.str.hypenate(yAxis.namespace(),'object-container')) // xLock = chartSel.node().getBBox().width - chart.spaceX() - chartSel.node().getBBox().x // yLock = chartSel.node().getBBox().height - chart.spaceY() @@ -197,9 +196,9 @@ export function plotZoom( chart, xAxis, yAxis ) { // bhm.selection().node().getBBox().width - bhm.spaceX() - var chartObjTrans = getTranslation(chartObjSel.attr('transform')) - var xAxisObjTrans = getTranslation(xAxisObjSel.attr('transform')) - var yAxisObjTrans = getTranslation(yAxisObjSel.attr('transform')) + var chartObjTrans = utils.sel.getTranslation(chartObjSel.attr('transform')) + var xAxisObjTrans = utils.sel.getTranslation(xAxisObjSel.attr('transform')) + var yAxisObjTrans = utils.sel.getTranslation(yAxisObjSel.attr('transform')) var x = horizontalQ ? transform.applyX(chartObjTrans[0]) : 0 @@ -225,9 +224,9 @@ export function plotZoom( chart, xAxis, yAxis ) { if (orient == 'horizontal') {horizontalQ = true; verticalQ = false;} if (orient == 'vertical') {verticalQ = true; horizontalQ = false;} - var chartObjSel = chartSel.select('.'+hypenate(chart.namespace(),'object-container')) - var xAxisObjSel = xAxisSel.select('.'+hypenate(xAxis.namespace(),'object-container')) - var yAxisObjSel = yAxisSel.select('.'+hypenate(yAxis.namespace(),'object-container')) + var chartObjSel = chartSel.select('.'+utils.str.hypenate(chart.namespace(),'object-container')) + var xAxisObjSel = xAxisSel.select('.'+utils.str.hypenate(xAxis.namespace(),'object-container')) + var yAxisObjSel = yAxisSel.select('.'+utils.str.hypenate(yAxis.namespace(),'object-container')) chartObjSel.attr('transform', 'translate('+0+','+0+')') xAxisObjSel.attr('transform', 'translate('+0+','+0+')') yAxisObjSel.attr('transform', 'translate('+0+','+0+')') diff --git a/src/scripts/modules/axis.js b/src/modules/axis.js similarity index 91% rename from src/scripts/modules/axis.js rename to src/modules/axis.js index e2dae61..c3f3d1f 100644 --- a/src/scripts/modules/axis.js +++ b/src/modules/axis.js @@ -1,12 +1,5 @@ -import { - hypenate, safeSelect, extractViolinValues, - tickRange, modifyHexidecimalColorLuminance, truncateText, - truncateString, - round -} from './helpers'; -import {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer} from './utils'; -import {unique, hasQ, flatten} from './array-functions'; -import {groupingSpacer} from './grouping-spacer'; +import utils from './utils/index.js' +import groupingSpacer from './grouping-spacer'; /******************************************************************************* ** ** ** ** @@ -24,7 +17,7 @@ import {groupingSpacer} from './grouping-spacer'; * @namespace axis * @returns {function} axis */ -export function axis ( selection ) { +export default function axis ( selection ) { var /** * The orientation of the axis @@ -780,7 +773,7 @@ export function axis ( selection ) { function axis () { // for convenience in handling orientation specific values - var horizontalQ = hasQ(['top', 'bottom', 'horizontal'], orient) ? true : false + var horizontalQ = utils.arr.hasQ(['top', 'bottom', 'horizontal'], orient) ? true : false var verticalQ = !horizontalQ // background cliping rectangle @@ -815,7 +808,7 @@ export function axis ( selection ) { } - var container = setupContainer( selection, namespace, bgcpRect, backgroundFill ); + var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); // defaults for text-anchor and text rotation if (orient == 'top') { @@ -853,12 +846,12 @@ export function axis ( selection ) { : (grouping == undefined) ? (numberOfTicks != undefined) // ? (tickValues.length < numberOfTicks) - ? (tickRange(...d3.extent(tickValues), numberOfTicks)) + ? (utils.math.tickRange(...d3.extent(tickValues), numberOfTicks)) : tickValues : grouping - var flatTickData = flatten(tickData) + var flatTickData = utils.arr.flatten(tickData) var numberOfObjects = flatTickData.length var space = horizontalQ ? spaceX : spaceY var extent = d3.extent(flatTickData) @@ -882,16 +875,16 @@ export function axis ( selection ) { // calculate object size if needed objectSize = (objectSize == undefined) - ? calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ) + ? utils.math.calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ) : objectSize // calculate spacer size if needed spacerSize = (spacerSize == undefined) - ? calculateWidthOfSpacer(flatTickData, space, objectSize, numberOfObjects, objectSpacer, overflowQ) + ? utils.math.calculateWidthOfSpacer(flatTickData, space, objectSize, numberOfObjects, objectSpacer, overflowQ) : spacerSize - var objClass = hypenate(namespace, categoricalQ ? objectClass+'-categorical' : objectClass) + var objClass = utils.str.hypenate(namespace, categoricalQ ? objectClass+'-categorical' : objectClass) var spacerFunction = groupingSpacer() .horizontalQ(horizontalQ).scale(scale).moveby((categoricalQ?'category':'scale')).numberOfObjects(numberOfObjects) @@ -956,11 +949,11 @@ export function axis ( selection ) { - var labelNameGroup = safeSelect(selection, 'g', hypenate(namespace,'axis-name')) + var labelNameGroup = utils.sel.safeSelect(selection, 'g', utils.str.hypenate(namespace,'axis-name')) - var labelElement = safeSelect(labelNameGroup, 'text', hypenate(namespace, 'name')) + var labelElement = utils.sel.safeSelect(labelNameGroup, 'text', utils.str.hypenate(namespace, 'name')) if (labelElement != undefined) { labelElement.text(label) @@ -1008,8 +1001,8 @@ export function axis ( selection ) { to use clip path to make things fit in fixed size. Have yet got this to work nicely. */ // var defs = d3.select(container.node().parentNode).select('defs') - // var tickLabelClipPath = safeSelect(defs, 'clipPath', hypenate(namespace,'tick-label-clip-path')).attr('id', hypenate(namespace,'tick-label-clip-path')) - // var tickLabelClipPathRect = safeSelect(tickLabelClipPath, 'rect', hypenate(namespace,'tick-label-clip-path-rect')) + // var tickLabelClipPath = utils.sel.safeSelect(defs, 'clipPath', utils.str.hypenate(namespace,'tick-label-clip-path')).attr('id', utils.str.hypenate(namespace,'tick-label-clip-path')) + // var tickLabelClipPathRect = utils.sel.safeSelect(tickLabelClipPath, 'rect', utils.str.hypenate(namespace,'tick-label-clip-path-rect')) // .attr('x', 0) // .attr('y', 0) // .attr('width', function(d, i){ @@ -1027,7 +1020,7 @@ export function axis ( selection ) { var that = d3.select(this).style('opacity', 1) // make and move tick - var tick = safeSelect(that, 'line', hypenate(namespace,'tick')) + var tick = utils.sel.safeSelect(that, 'line', utils.str.hypenate(namespace,'tick')) .attr("x1", 0) .attr("x2", horizontalQ ? 0 : orient == "left" ? -tickLength : tickLength) .attr("y1", 0) @@ -1043,15 +1036,15 @@ export function axis ( selection ) { }) // make and move label - var label = safeSelect(that, 'text', hypenate(namespace,'label')) + var label = utils.sel.safeSelect(that, 'text', utils.str.hypenate(namespace,'label')) .text(function(d, i){ - var s = typeof d == 'number' ? round(d, roundTo) : d - s = truncateString(String(s), (horizontalQ ? spaceY : spaceX) - tickLength-tickLabelMargin-tickTickLabelSpacer, tickLabelFontSize * 0.45) + var s = typeof d == 'number' ? utils.math.round(d, roundTo) : d + s = utils.str.truncateString(String(s), (horizontalQ ? spaceY : spaceX) - tickLength-tickLabelMargin-tickTickLabelSpacer, tickLabelFontSize * 0.45) return s }) .attr('font-size', tickLabelFontSize) .attr('text-anchor', tickLabelTextAnchor) - // truncateText(label, label.text(), orient, tickLength, horizontalQ ? spaceY : spaceX, overflowQ) + // utils.str.truncateText(label, label.text(), orient, tickLength, horizontalQ ? spaceY : spaceX, overflowQ) label.attr('transform', function(d, i) { var @@ -1097,11 +1090,11 @@ export function axis ( selection ) { .on('mousemove', labelHover) .on('mouseout', labelHoverOff) .on('click', tickLabelOnClick) - // .attr('clip-path', 'url(#'+hypenate(namespace,'tick-label-clip-path')+')') + // .attr('clip-path', 'url(#'+utils.str.hypenate(namespace,'tick-label-clip-path')+')') // add guidlines as needed if (guideLinesQ) { - var gline = safeSelect(that, 'line', hypenate(namespace, 'guideline')) + var gline = utils.sel.safeSelect(that, 'line', utils.str.hypenate(namespace, 'guideline')) .transition().duration(transitionDuration).ease(easeFunc) .attr("x1", 0) .attr("x2", horizontalQ ? 0 : orient == "left" ? guidelineSpace : -guidelineSpace) @@ -1114,15 +1107,15 @@ export function axis ( selection ) { t = 'translate('+x+','+y+')' return t }) - } else { that.select('line.'+hypenate(namespace, 'guideline')).remove() } + } else { that.select('line.'+utils.str.hypenate(namespace, 'guideline')).remove() } }) // apply alternating guidline thickness if (guideLinesQ) { - container.selectAll('.'+hypenate(namespace,'guideline')) + container.selectAll('.'+utils.str.hypenate(namespace,'guideline')) .attr('stroke', function(d, i){ - if (i % 2 == 0) { return modifyHexidecimalColorLuminance(guideLineStroke, 0.8) } + if (i % 2 == 0) { return utils.color.modifyHexidecimalColorLuminance(guideLineStroke, 0.8) } return guideLineStroke }) .attr('stroke-width', function(d, i){ @@ -1136,7 +1129,7 @@ export function axis ( selection ) { /*************************************************************************** ** Make the line of the axis ***************************************************************************/ - var line = safeSelect(selection, 'path', hypenate(namespace,'line')) + var line = utils.sel.safeSelect(selection, 'path', utils.str.hypenate(namespace,'line')) // .attr('x1', 0) // .attr('x2', horizontalQ ? spaceX : 0) // .attr('y1', 0) @@ -1156,21 +1149,21 @@ export function axis ( selection ) { // hover of label show full text label in case it is truncated function labelHover(d, i){ var t = d3.select(this).style('fill', 'red') - d3.select(t.node().parentNode).select("line."+hypenate(namespace,'tick')) + d3.select(t.node().parentNode).select("line."+utils.str.hypenate(namespace,'tick')) .attr("stroke", 'red') .attr("stroke-width", tickStrokeWidth*2) if (guideLinesQ) { - d3.select(t.node().parentNode).select('line.'+hypenate(namespace, 'guideline')) + d3.select(t.node().parentNode).select('line.'+utils.str.hypenate(namespace, 'guideline')) .attr('stroke', 'red') .attr('stroke-width', guideLineStrokeWidth*2) } - var s = typeof d == 'number' ? round(d, roundTo) : d + var s = typeof d == 'number' ? utils.math.round(d, roundTo) : d var m = d3.mouse(d3.select('html').node()) - var div = safeSelect(d3.select('body'), 'div', hypenate(namespace,'guideline-tooltip')) - .attr('id', hypenate(namespace,'guideline-tooltip')) + var div = utils.sel.safeSelect(d3.select('body'), 'div', utils.str.hypenate(namespace,'guideline-tooltip')) + .attr('id', utils.str.hypenate(namespace,'guideline-tooltip')) .style('position', 'absolute') .style('left', (d3.event.pageX+15)+'px') .style('top', (d3.event.pageY+15)+'px') @@ -1187,7 +1180,7 @@ export function axis ( selection ) { .style('border-style', 'solid') .style('border-width', 2) - var text = safeSelect(div, 'div') + var text = utils.sel.safeSelect(div, 'div') .text(tickLabelOnHoverFunc(s, i)) .style('color', 'black') .style('align-self', 'center') @@ -1200,15 +1193,15 @@ export function axis ( selection ) { function labelHoverOff(d, i){ var t = d3.select(this).style('fill', 'black') - d3.select(t.node().parentNode).select("line."+hypenate(namespace,'tick')) + d3.select(t.node().parentNode).select("line."+utils.str.hypenate(namespace,'tick')) .attr("stroke", tickStroke) .attr("stroke-width", tickStrokeWidth) if (guideLinesQ) { - var gline = d3.select(t.node().parentNode).select('line.'+hypenate(namespace, 'guideline')) + var gline = d3.select(t.node().parentNode).select('line.'+utils.str.hypenate(namespace, 'guideline')) var minorQ = gline.attr('minor') gline.attr('stroke', function(d, ii){ - if (minorQ == 'true') { return modifyHexidecimalColorLuminance(guideLineStroke, 0.8) } + if (minorQ == 'true') { return utils.color.modifyHexidecimalColorLuminance(guideLineStroke, 0.8) } return guideLineStroke }) .attr('stroke-width', function(d, ii){ @@ -1216,7 +1209,7 @@ export function axis ( selection ) { return guideLineStrokeWidth }) } - d3.select("#"+hypenate(namespace,'guideline-tooltip')).remove() + d3.select("#"+utils.str.hypenate(namespace,'guideline-tooltip')).remove() } diff --git a/src/scripts/modules/bar.js b/src/modules/charts/bar.js similarity index 96% rename from src/scripts/modules/bar.js rename to src/modules/charts/bar.js index 6e75c75..82318b9 100644 --- a/src/scripts/modules/bar.js +++ b/src/modules/charts/bar.js @@ -1,9 +1,7 @@ -import {hypenate, safeSelect} from './helpers'; -import {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer} from './utils'; -import {unique, flatten} from './array-functions'; -import {groupingSpacer} from './grouping-spacer'; -import {colorFunction as CF} from './color-function'; -import {tooltip as TTip} from './tooltip'; +import utils from '../utils/index.js' +import groupingSpacer from '../grouping-spacer'; +import CF from '../color-function'; +import TTip from '../tooltip'; /******************************************************************************* ** ** ** ** @@ -21,7 +19,7 @@ import {tooltip as TTip} from './tooltip'; * @namespace bar * @returns {function} bar */ -export function bar ( selection ) { +export default function bar ( selection ) { /* Assumes that data is list an object. @@ -503,7 +501,7 @@ export function bar ( selection ) { // background cliping rectangle var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY} - var container = setupContainer( selection, namespace, bgcpRect, backgroundFill ); + var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); // to prevent re-calculation and getters to be passed to axes barKeys = d3.keys(data) @@ -512,7 +510,7 @@ export function bar ( selection ) { // if grouping is undefined sort barKeys by sortingFunction var ordered = (grouping == undefined) ? barKeys.sort(sortingFunction) : grouping // ordered might be nested depending on grouping - barKeys = flatten(ordered) + barKeys = utils.arr.flatten(ordered) var numberOfObjects = barKeys.length var extent = [Math.min(...barValues) - domainPadding,Math.max(...barValues) + domainPadding]; @@ -530,12 +528,12 @@ export function bar ( selection ) { var space = horizontalQ ? spaceX : spaceY // calculate object size objectSize = (objectSize == undefined) - ? calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ) + ? utils.math.calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ) : objectSize // calculate spacer size if needed spacerSize = (spacerSize == undefined) - ? calculateWidthOfSpacer(barKeys, space, objectSize, numberOfObjects, objectSpacer, overflowQ) + ? utils.math.calculateWidthOfSpacer(barKeys, space, objectSize, numberOfObjects, objectSpacer, overflowQ) : spacerSize // make the nested groups var spacerFunction = groupingSpacer() @@ -602,7 +600,7 @@ export function bar ( selection ) { strokeColor = colorFunction(key, value, i, 'stroke') - var bar = safeSelect(t, 'rect', 'bar-rect') + var bar = utils.sel.safeSelect(t, 'rect', 'bar-rect') if (bar.attr('transform') == undefined) { bar.attr('transform', function(d, i) { diff --git a/src/scripts/modules/box-whisker.js b/src/modules/charts/box-whisker.js similarity index 94% rename from src/scripts/modules/box-whisker.js rename to src/modules/charts/box-whisker.js index 03a0842..55e0638 100644 --- a/src/scripts/modules/box-whisker.js +++ b/src/modules/charts/box-whisker.js @@ -1,9 +1,7 @@ -import {hypenate, safeSelect} from './helpers'; -import {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer, whiskerPath} from './utils'; -import {unique, hasQ, flatten} from './array-functions'; -import {groupingSpacer} from './grouping-spacer'; -import {colorFunction as CF} from './color-function'; -import {tooltip as TTip} from './tooltip'; +import utils from '../utils/index.js'; +import groupingSpacer from '../grouping-spacer'; +import CF from '../color-function'; +import TTip from '../tooltip'; /******************************************************************************* ** ** ** ** @@ -550,12 +548,12 @@ export function boxwhisker( selection ) { // background cliping rectangle var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY} - var container = setupContainer( selection, namespace, bgcpRect, backgroundFill ); + var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); // if grouping is undefined sort keys by sorting funct var ordered = (grouping == undefined) ? d3.keys(data).sort(sortingFunction) : grouping // to prevent re-calculation and getters to be passed to axes - boxKeys = flatten(ordered) + boxKeys = utils.arr.flatten(ordered) boxValues = boxKeys.map(valueExtractor) @@ -569,9 +567,9 @@ export function boxwhisker( selection ) { scale.domain(extent).range(horizontalQ ? [0,spaceY] : [spaceX, 0]) var space = horizontalQ ? spaceX : spaceY // calculate object size - objectSize = calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ) + objectSize = utils.math.calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ) // calculate spacer size if needed - spacerSize = calculateWidthOfSpacer(boxKeys, space, objectSize, numberOfObjects, objectSpacer, overflowQ) + spacerSize = utils.math.calculateWidthOfSpacer(boxKeys, space, objectSize, numberOfObjects, objectSpacer, overflowQ) // make the nested groups var spacerFunction = groupingSpacer() .horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects) @@ -584,7 +582,7 @@ export function boxwhisker( selection ) { var parentIndexArray = [] container.selectAll('g:not(.to-remove).'+objectClass) - .each(function(d, i){if (hasQ(boxKeys, d)){ parentIndexArray.push(Number(d3.select(this).attr('parent-index')))}}) + .each(function(d, i){if (utils.arr.hasQ(boxKeys, d)){ parentIndexArray.push(Number(d3.select(this).attr('parent-index')))}}) @@ -611,13 +609,13 @@ export function boxwhisker( selection ) { var - whisk = safeSelect(t, 'g', 'whisker'), - uWhisk = safeSelect(whisk, 'path', 'upper'), - lWhisk = safeSelect(whisk, 'path', 'lower'), - quart = safeSelect(t, 'g', 'quartile'), - uQuart = safeSelect(quart, 'rect', 'upper'), - lQuart = safeSelect(quart, 'rect', 'lower'), - mQuart = safeSelect(quart, 'circle', 'median') + whisk = utils.sel.safeSelect(t, 'g', 'whisker'), + uWhisk = utils.sel.safeSelect(whisk, 'path', 'upper'), + lWhisk = utils.sel.safeSelect(whisk, 'path', 'lower'), + quart = utils.sel.safeSelect(t, 'g', 'quartile'), + uQuart = utils.sel.safeSelect(quart, 'rect', 'upper'), + lQuart = utils.sel.safeSelect(quart, 'rect', 'lower'), + mQuart = utils.sel.safeSelect(quart, 'circle', 'median') // set upper quartile (q3) @@ -678,7 +676,7 @@ export function boxwhisker( selection ) { y = 0, h = horizontalQ ? scale(q1) - scale(q0) : objectSize, w = verticalQ ? scale(q1) - scale(q0) : objectSize - return whiskerPath(dir, x, y, w, h, whiskerWidthPercent, orient) + return utils.paths.whiskerPath(dir, x, y, w, h, whiskerWidthPercent, orient) }) .attr('transform', function(d, i){ var @@ -699,7 +697,7 @@ export function boxwhisker( selection ) { y = 0, h = horizontalQ ? scale(q4) - scale(q3) : objectSize, w = verticalQ ? scale(q4) - scale(q3) : objectSize - return whiskerPath(dir, x, y, w, h, whiskerWidthPercent, orient) + return utils.paths.whiskerPath(dir, x, y, w, h, whiskerWidthPercent, orient) }) .attr('transform', function(d, i){ var diff --git a/src/scripts/modules/bubble-heatmap.js b/src/modules/charts/bubble-heatmap.js similarity index 91% rename from src/scripts/modules/bubble-heatmap.js rename to src/modules/charts/bubble-heatmap.js index 8d4707e..906e05c 100644 --- a/src/scripts/modules/bubble-heatmap.js +++ b/src/modules/charts/bubble-heatmap.js @@ -1,9 +1,7 @@ -import {hypenate, safeSelect} from './helpers'; -import {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer, log} from './utils'; -import {unique} from './array-functions'; -import {groupingSpacer} from './grouping-spacer'; -import {colorFunction as CF} from './color-function'; -import {tooltip as TTip} from './tooltip'; +import utils from '../utils/index.js' +import groupingSpacer from '../grouping-spacer'; +import CF from '../color-function'; +import TTip from '../tooltip'; /** @@ -15,7 +13,7 @@ import {tooltip as TTip} from './tooltip'; * @namespace bubbleHeatmap * @returns {function} bubbleHeatmap */ -function bubbleHeatmap( selection ) { +export default function bubbleHeatmap( selection ) { var /** * Data to plot. Assumed to be a object, where each key corresponds to a cell @@ -220,7 +218,7 @@ function bubbleHeatmap( selection ) { */ cellKeys, /** - * Stores the list of unique xValues + * Stores the list of utils.arr.unique xValues * Calculated after bubbleHeatmap called. * @param {string[]} [xValues=undefined] * @memberof bubbleHeatmap# @@ -228,7 +226,7 @@ function bubbleHeatmap( selection ) { */ xValues, /** - * Stores the list of unique yValues + * Stores the list of utils.arr.unique yValues * Calculated after bubbleHeatmap called. * @param {string[]} [yValues=undefined] * @memberof bubbleHeatmap# @@ -236,7 +234,7 @@ function bubbleHeatmap( selection ) { */ yValues, /** - * Stores the list of unique rValues + * Stores the list of utils.arr.unique rValues * Calculated after bubbleHeatmap called. * @param {string[]} [rValues=undefined] * @memberof bubbleHeatmap# @@ -244,7 +242,7 @@ function bubbleHeatmap( selection ) { */ rValues, /** - * Stores the list of unique vValues + * Stores the list of utils.arr.unique vValues * Calculated after bubbleHeatmap called. * @param {string[]} [vValues=undefined] * @memberof bubbleHeatmap# @@ -673,19 +671,19 @@ function bubbleHeatmap( selection ) { function bhm() { var horizontalQ = true; // no orientation in heatmaps. Aids as placeholder in functions that take orient as a parameter var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY} - var container = setupContainer( selection, namespace, bgcpRect, backgroundFill ); + var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); cellKeys = d3.keys(data); cellKeys.sort(function(a, b){ return xKeySortingFunction(a, b) || yKeySortingFunction(a, b) }) - log('bubbleHeatmap', 'cells are sorted by', cellKeys) + utils.con.log('bubbleHeatmap', 'cells are sorted by', cellKeys) - xValues = unique(cellKeys.map(xExtractor)); - yValues = unique(cellKeys.map(yExtractor)); - rValues = unique(cellKeys.map(rExtractor)); - vValues = unique(cellKeys.map(vExtractor)); - log('bubbleHeatmap', 'x and y keys are', {x: xValues, y:yValues}) + xValues = utils.arr.unique(cellKeys.map(xExtractor)); + yValues = utils.arr.unique(cellKeys.map(yExtractor)); + rValues = utils.arr.unique(cellKeys.map(rExtractor)); + vValues = utils.arr.unique(cellKeys.map(vExtractor)); + utils.con.log('bubbleHeatmap', 'x and y keys are', {x: xValues, y:yValues}) var xDim = xValues.length, yDim = yValues.length; @@ -694,11 +692,11 @@ function bubbleHeatmap( selection ) { var extent = [Math.min(...rValues) - domainPadding,Math.max(...rValues) + domainPadding]; - ySize = calculateWidthOfObject(spaceY, yDim, minObjectSize, maxObjectSize, objectSpacer, overflowQ) - xSize = calculateWidthOfObject(spaceX, xDim, minObjectSize, maxObjectSize, objectSpacer, overflowQ) - ySpacerSize = calculateWidthOfSpacer(yValues, spaceY, ySize, yDim, objectSpacer, overflowQ) - xSpacerSize = calculateWidthOfSpacer(xValues, spaceX, xSize, xDim, objectSpacer, overflowQ) - log('bubbleHeatmap', 'size of', {x: xSize, y: ySize}) + ySize = utils.math.calculateWidthOfObject(spaceY, yDim, minObjectSize, maxObjectSize, objectSpacer, overflowQ) + xSize = utils.math.calculateWidthOfObject(spaceX, xDim, minObjectSize, maxObjectSize, objectSpacer, overflowQ) + ySpacerSize = utils.math.calculateWidthOfSpacer(yValues, spaceY, ySize, yDim, objectSpacer, overflowQ) + xSpacerSize = utils.math.calculateWidthOfSpacer(xValues, spaceX, xSize, xDim, objectSpacer, overflowQ) + utils.con.log('bubbleHeatmap', 'size of', {x: xSize, y: ySize}) scale.domain(extent).range([Math.min(minObjectSize/2, Math.min(ySize, xSize)/2), Math.min(ySize, xSize)/2]) @@ -706,7 +704,7 @@ function bubbleHeatmap( selection ) { var ySpacer = groupingSpacer() .horizontalQ(false) .moveby('category').numberOfObjects(yDim) - .objectClass(hypenate(objectClass, 'row')) + .objectClass(utils.str.hypenate(objectClass, 'row')) .objectSize(ySize).spacerSize(ySpacerSize) .transitionDuration(transitionDuration).easeFunc(easeFunc) .namespace('row') @@ -720,7 +718,7 @@ function bubbleHeatmap( selection ) { ySpacer(container, yValues, 0) - container.selectAll('g.'+hypenate(objectClass, 'row')) + container.selectAll('g.'+utils.str.hypenate(objectClass, 'row')) .each(function(d, i){ xSpacer(d3.select(this), xValues, 0) }) @@ -735,7 +733,7 @@ function bubbleHeatmap( selection ) { : colorFunction.dataExtent([0, Math.max(...vValues)]) cells.each(function(key, i) { - log('bubbleHeatmap', 'each cell', {key: key, index: i, node: d3.select(this).node()}) + utils.con.log('bubbleHeatmap', 'each cell', {key: key, index: i, node: d3.select(this).node()}) var t = d3.select(this), currentData = data[key], @@ -745,9 +743,9 @@ function bubbleHeatmap( selection ) { fillColor = colorFunction(key, value, i, 'fill'), // prevent duplicate computation strokeColor = colorFunction(key, value, i, 'stroke') - log('bubbleHeatmap', 'radius',{radius: radius, scaled: scale(radius), extent: extent, range:scale.range()}) + utils.con.log('bubbleHeatmap', 'radius',{radius: radius, scaled: scale(radius), extent: extent, range:scale.range()}) - var c = safeSelect(t, 'circle', hypenate(objectClass,'circle')) + var c = utils.sel.safeSelect(t, 'circle', utils.str.hypenate(objectClass,'circle')) c.attr('cx', xSize / 2) .attr('cy', ySize / 2 ) .attr('r', scale(radius)) @@ -757,10 +755,10 @@ function bubbleHeatmap( selection ) { }) - tooltip.selection(cells.selectAll('circle.'+hypenate(objectClass, 'circle'))) + tooltip.selection(cells.selectAll('circle.'+utils.str.hypenate(objectClass, 'circle'))) .data(data) // .keys(['r', 'v']) - // .header(function(d, i){return hypenate(data[d][xKey], data[d][yKey]) }) + // .header(function(d, i){return utils.str.hypenate(data[d][xKey], data[d][yKey]) }) tooltip() @@ -769,5 +767,3 @@ function bubbleHeatmap( selection ) { return bhm; } - -export {bubbleHeatmap} diff --git a/src/scripts/modules/heatmap.js b/src/modules/charts/heatmap.js similarity index 92% rename from src/scripts/modules/heatmap.js rename to src/modules/charts/heatmap.js index a55c7f6..57777c0 100644 --- a/src/scripts/modules/heatmap.js +++ b/src/modules/charts/heatmap.js @@ -1,9 +1,7 @@ -import {hypenate, safeSelect} from './helpers'; -import {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer, log} from './utils'; -import {unique} from './array-functions'; -import {groupingSpacer} from './grouping-spacer'; -import {colorFunction as CF} from './color-function'; -import {tooltip as TTip} from './tooltip'; +import utils from '../utils/index.js' +import groupingSpacer from '../grouping-spacer'; +import CF from '../color-function'; +import TTip from '../tooltip'; /** @@ -15,7 +13,7 @@ import {tooltip as TTip} from './tooltip'; * @namespace heatmap * @returns {function} heatmap */ -function heatmap( selection ) { +export default function heatmap( selection ) { var /** * Data to plot. Assumed to be a object, where each key corresponds to a cell @@ -204,7 +202,7 @@ function heatmap( selection ) { */ cellKeys, /** - * Stores the list of unique xValues + * Stores the list of utils.arr.unique xValues * Calculated after heatmap called. * @param {string[]} [xValues=undefined] * @memberof heatmap# @@ -212,7 +210,7 @@ function heatmap( selection ) { */ xValues, /** - * Stores the list of unique yValues + * Stores the list of utils.arr.unique yValues * Calculated after heatmap called. * @param {string[]} [yValues=undefined] * @memberof heatmap# @@ -220,7 +218,7 @@ function heatmap( selection ) { */ yValues, /** - * Stores the list of unique vValues + * Stores the list of utils.arr.unique vValues * Calculated after heatmap called. * @param {string[]} [vValues=undefined] * @memberof heatmap# @@ -619,29 +617,29 @@ function heatmap( selection ) { function hm() { var horizontalQ = true; // no orientation in heatmaps. Aids as placeholder in functions that take orient as a parameter var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY} - var container = setupContainer( selection, namespace, bgcpRect, backgroundFill ); + var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); cellKeys = d3.keys(data); - xValues = unique(cellKeys.map(xExtractor)); - yValues = unique(cellKeys.map(yExtractor)); - vValues = unique(cellKeys.map(vExtractor)); + xValues = utils.arr.unique(cellKeys.map(xExtractor)); + yValues = utils.arr.unique(cellKeys.map(yExtractor)); + vValues = utils.arr.unique(cellKeys.map(vExtractor)); cellKeys.sort(function(a, b){ return xKeySortingFunction(a, b) || yKeySortingFunction(a, b) }) - log('heatmap', 'cells are sorted by', cellKeys) + utils.con.log('heatmap', 'cells are sorted by', cellKeys) - log('heatmap', 'x and y keys are', {x: xValues, y:yValues}) + utils.con.log('heatmap', 'x and y keys are', {x: xValues, y:yValues}) var xDim = xValues.length, yDim = yValues.length; - ySize = calculateWidthOfObject(spaceY, yDim, yMinObjectSize, yMaxObjectSize, objectSpacer, overflowQ) - xSize = calculateWidthOfObject(spaceX, xDim, xMinObjectSize, xMaxObjectSize, objectSpacer, overflowQ) - ySpacerSize = calculateWidthOfSpacer(yValues, spaceY, ySize, yDim, objectSpacer, overflowQ) - xSpacerSize = calculateWidthOfSpacer(xValues, spaceX, xSize, xDim, objectSpacer, overflowQ) + ySize = utils.math.calculateWidthOfObject(spaceY, yDim, yMinObjectSize, yMaxObjectSize, objectSpacer, overflowQ) + xSize = utils.math.calculateWidthOfObject(spaceX, xDim, xMinObjectSize, xMaxObjectSize, objectSpacer, overflowQ) + ySpacerSize = utils.math.calculateWidthOfSpacer(yValues, spaceY, ySize, yDim, objectSpacer, overflowQ) + xSpacerSize = utils.math.calculateWidthOfSpacer(xValues, spaceX, xSize, xDim, objectSpacer, overflowQ) // console.table({ // x:{ // object: xSize, @@ -655,7 +653,7 @@ function heatmap( selection ) { // } // // }) - log('heatmap', 'size of', {x: xSize, y: ySize}) + utils.con.log('heatmap', 'size of', {x: xSize, y: ySize}) @@ -663,7 +661,7 @@ function heatmap( selection ) { .horizontalQ(false) .moveby('category') .numberOfObjects(yDim) - .objectClass(hypenate(objectClass, 'row')) + .objectClass(utils.str.hypenate(objectClass, 'row')) .objectSize(ySize + ySpacerSize) .spacerSize(0) .transitionDuration(transitionDuration) @@ -682,7 +680,7 @@ function heatmap( selection ) { ySpacer(container, yValues, 0) - container.selectAll('g.'+hypenate(objectClass, 'row')) + container.selectAll('g.'+utils.str.hypenate(objectClass, 'row')) .each(function(d, i){ xSpacer(d3.select(this), xValues, 0) }) var cells = container.selectAll('g:not(.to-remove).'+objectClass) @@ -725,7 +723,7 @@ function heatmap( selection ) { : colorFunction.dataExtent([0, Math.max(...vValues)]) cells.each(function(key, i) { - log('heatmap', 'each cell', {key: key, index: i, node: d3.select(this).node()}) + utils.con.log('heatmap', 'each cell', {key: key, index: i, node: d3.select(this).node()}) var t = d3.select(this) if (key == undefined) {return} @@ -735,7 +733,7 @@ function heatmap( selection ) { fillColor = colorFunction(key, value, i, 'fill'), // prevent duplicate computation strokeColor = colorFunction(key, value, i, 'stroke') - var c = safeSelect(t, 'rect', hypenate(objectClass,'rect')) + var c = utils.sel.safeSelect(t, 'rect', utils.str.hypenate(objectClass,'rect')) c.attr('width', xSize + xSpacerSize - objectStrokeWidth) .attr('height', ySize + ySpacerSize - objectStrokeWidth) .attr('fill', fillColor) @@ -746,10 +744,10 @@ function heatmap( selection ) { }) - tooltip.selection(cells.selectAll('rect.'+hypenate(objectClass, 'rect'))) + tooltip.selection(cells.selectAll('rect.'+utils.str.hypenate(objectClass, 'rect'))) .data(data) // .keys(['r', 'v']) - // .header(function(d, i){return hypenate(data[d][xKey], data[d][yKey]) }) + // .header(function(d, i){return utils.str.hypenate(data[d][xKey], data[d][yKey]) }) tooltip() @@ -758,5 +756,3 @@ function heatmap( selection ) { return hm; } - -export {heatmap} diff --git a/src/modules/charts/index.js b/src/modules/charts/index.js new file mode 100644 index 0000000..05d8b08 --- /dev/null +++ b/src/modules/charts/index.js @@ -0,0 +1,14 @@ +import scatter from './scatter' +import bar from './bar' +import bubble from './bubble-heatmap' +import heatmap from './heatmap' +import violin from './violin' +import {neededViolinValues} from './violin' +import upset from './upset' + +let charts = { + scatter, bar, bubble, heatmap, violin, neededViolinValues, upset +} + +export {scatter, bar, bubble, heatmap, violin, neededViolinValues, upset} +export default charts diff --git a/src/scripts/modules/scatter.js b/src/modules/charts/scatter.js similarity index 97% rename from src/scripts/modules/scatter.js rename to src/modules/charts/scatter.js index 2dcb81b..07aea69 100644 --- a/src/scripts/modules/scatter.js +++ b/src/modules/charts/scatter.js @@ -1,9 +1,6 @@ -import {hypenate, safeSelect} from './helpers'; -import {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer} from './utils'; -import {unique, flatten} from './array-functions'; -import {groupingSpacer} from './grouping-spacer'; -import {colorFunction as CF} from './color-function'; -import {tooltip as TTip} from './tooltip'; +import utils from '../utils/index.js' +import CF from '../color-function'; +import TTip from '../tooltip'; /******************************************************************************* ** ** ** ** @@ -20,7 +17,7 @@ import {tooltip as TTip} from './tooltip'; * @namespace scatter * @returns {function} scatter */ -export function scatter ( selection ) { +export default function scatter ( selection ) { var /** @@ -500,7 +497,7 @@ export function scatter ( selection ) { function scatter() { // background cliping rectangle var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY} - var container = setupContainer( selection, namespace, bgcpRect, backgroundFill ); + var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); pointKeys = d3.keys(data) diff --git a/src/scripts/modules/upset.js b/src/modules/charts/upset.js similarity index 84% rename from src/scripts/modules/upset.js rename to src/modules/charts/upset.js index 506f6e3..aedfde4 100644 --- a/src/scripts/modules/upset.js +++ b/src/modules/charts/upset.js @@ -1,10 +1,8 @@ -import {hypenate, safeSelect} from './helpers'; -import {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer, log} from './utils'; -import {unique, flatten} from './array-functions'; -import {groupingSpacer} from './grouping-spacer'; -import {colorFunction as CF} from './color-function'; -import {tooltip as TTip} from './tooltip'; -export function upset ( selection ) { +import utils from '../utils/index.js' +import groupingSpacer from '../grouping-spacer'; +import CF from '../color-function'; +import TTip from '../tooltip'; +export default function upset ( selection ) { var data, orient='horizontal', @@ -88,12 +86,12 @@ export function upset ( selection ) { // background cliping rectangle var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY} - var container = setupContainer( selection, namespace, bgcpRect, backgroundFill ); + var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); cellKeys = d3.keys(data) - setValues = unique(cellKeys.map(setExtractor)).sort() - intersectionValues = unique(cellKeys.map(intersectionExtractor)).sort().sort(function(a, b){ + setValues = utils.arr.unique(cellKeys.map(setExtractor)).sort() + intersectionValues = utils.arr.unique(cellKeys.map(intersectionExtractor)).sort().sort(function(a, b){ return a.split(';').length - b.split(';').length }) @@ -112,13 +110,13 @@ export function upset ( selection ) { xDim = horizontalQ ? xValues.length : yValues.length, yDim = horizontalQ ? yValues.length : xValues.length - // console.log(xValues, yValues) + // console.utils.con.log(xValues, yValues) - xObjectSize = calculateWidthOfObject(spaceX, xDim, minObjectSize, maxObjectSize, xObjectSpacer, overflowQ) - yObjectSize = calculateWidthOfObject(spaceY, yDim, minObjectSize, maxObjectSize, yObjectSpacer, overflowQ) - xSpacerSize = calculateWidthOfSpacer(xValues, spaceX, xObjectSize, xDim, xObjectSpacer, overflowQ) - ySpacerSize = calculateWidthOfSpacer(yValues, spaceY, yObjectSize, yDim, yObjectSpacer, overflowQ) + xObjectSize = utils.math.calculateWidthOfObject(spaceX, xDim, minObjectSize, maxObjectSize, xObjectSpacer, overflowQ) + yObjectSize = utils.math.calculateWidthOfObject(spaceY, yDim, minObjectSize, maxObjectSize, yObjectSpacer, overflowQ) + xSpacerSize = utils.math.calculateWidthOfSpacer(xValues, spaceX, xObjectSize, xDim, xObjectSpacer, overflowQ) + ySpacerSize = utils.math.calculateWidthOfSpacer(yValues, spaceY, yObjectSize, yDim, yObjectSpacer, overflowQ) var ySpacer = groupingSpacer() .horizontalQ(false) @@ -137,17 +135,17 @@ export function upset ( selection ) { if (verticalQ) { xSpacer.objectClass(objectClass) - ySpacer.namespace('across').objectClass(hypenate(objectClass, 'across')) + ySpacer.namespace('across').objectClass(utils.str.hypenate(objectClass, 'across')) ySpacer(container, yValues, 0) - container.selectAll('g.'+hypenate(objectClass, 'across')) + container.selectAll('g.'+utils.str.hypenate(objectClass, 'across')) .each(function(d, i){ xSpacer(d3.select(this), xValues, 0) }) } else { - xSpacer.namespace('across').objectClass(hypenate(objectClass, 'across')) + xSpacer.namespace('across').objectClass(utils.str.hypenate(objectClass, 'across')) ySpacer.objectClass(objectClass) xSpacer(container, xValues, 0) - container.selectAll('g.'+hypenate(objectClass, 'across')) + container.selectAll('g.'+utils.str.hypenate(objectClass, 'across')) .each(function(d, i){ ySpacer(d3.select(this), yValues, 0) }) } @@ -167,11 +165,11 @@ export function upset ( selection ) { // } else { // positionedCellKeys.push(lookupValue) // } - // console.log(i, j, lookupValue) + // console.utils.con.log(i, j, lookupValue) // } // } // - // // console.log(positionedCellKeys) + // // console.utils.con.log(positionedCellKeys) // // cells.data(positionedCellKeys); @@ -183,17 +181,17 @@ export function upset ( selection ) { if (key == undefined) {return } var currentData = data[key] - // console.log(key, currentData) + // console.utils.con.log(key, currentData) var set = setExtractor(key, i), intersection = intersectionExtractor(key, i) - // console.log(set, intersection) + // console.utils.con.log(set, intersection) t.classed(intersection, true) t.classed(set, true) - var c = safeSelect(t, 'circle', hypenate(objectClass,'circle')) + var c = utils.sel.safeSelect(t, 'circle', utils.str.hypenate(objectClass,'circle')) c.attr('cx', xObjectSize / 2) .attr('cy', yObjectSize / 2 ) .attr('r', radius == undefined ? Math.min(xObjectSize, yObjectSize) / 2 : radius) diff --git a/src/scripts/modules/violin.js b/src/modules/charts/violin.js similarity index 92% rename from src/scripts/modules/violin.js rename to src/modules/charts/violin.js index b6b7eea..564e493 100644 --- a/src/scripts/modules/violin.js +++ b/src/modules/charts/violin.js @@ -1,9 +1,6 @@ -import {hypenate, safeSelect, modifyHexidecimalColorLuminance, extractViolinValues, quartiles} from './helpers'; -import {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer} from './utils'; -import {unique, hasQ, flatten, whichBin} from './array-functions'; -import {groupingSpacer} from './grouping-spacer'; -import {colorFunction as CF} from './color-function'; -import {tooltip as TTip} from './tooltip'; +import groupingSpacer from '../grouping-spacer'; +import CF from '../color-function'; +import TTip from '../tooltip'; /******************************************************************************* ** ** @@ -22,7 +19,7 @@ import {tooltip as TTip} from './tooltip'; * @namespace violin * @returns {function} violin */ -export function violin( selection ) { +export default function violin( selection ) { var /** * Data to plot. Assumed to be a object, where each key corresponds to a violin @@ -156,9 +153,9 @@ export function violin( selection ) { */ pointColorFunc = function (d, type, base, min, max) { var minMaxHexScale = d3.scaleLinear().domain([min, max]).range([-0.25, 0.05]) - var scaledColor = modifyHexidecimalColorLuminance(base.replace('#', ''), minMaxHexScale(d)) + var scaledColor = utils.color.modifyHexidecimalColorLuminance(base.replace('#', ''), minMaxHexScale(d)) var mod = type == "stroke" ? 0 : 0.25 - return modifyHexidecimalColorLuminance(scaledColor.replace('#', ''), mod) + return utils.color.modifyHexidecimalColorLuminance(scaledColor.replace('#', ''), mod) }, /** @@ -213,12 +210,12 @@ export function violin( selection ) { easeFunc = d3.easeExp, /** - * The key containing the quartiles + * The key containing the utils.math.quartiles * @param {string} [quartilesKey=undefined] * @memberof violin# * @property */ - quartilesKey = "quartiles", + quartilesKey = "utils.math.quartiles", /** * The keys corresponding to each quartile * @param {string[]} [quartileKeys=["Q0", "Q1", "Q2", "Q3", "Q4"]] @@ -578,7 +575,7 @@ export function violin( selection ) { * @returns {violin | string} * @memberof violin * @property - * by default quartileKey = "quartiles" + * by default quartileKey = "utils.math.quartiles" */ violin.quartileKey = function(_) { return arguments.length ? (quartileKey = _, violin) : quartileKey; }; /** @@ -666,14 +663,14 @@ export function violin( selection ) { // background cliping rectangle var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY} - var container = setupContainer( selection, namespace, bgcpRect, backgroundFill ); + var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); // if grouping is undefined sort violinKeys by sortingFunction var ordered = (grouping == undefined) ? d3.keys(data).sort(sortingFunction) : grouping // console.log(ordered) - violinKeys = flatten(ordered) + violinKeys = utils.arr.flatten(ordered) var calcValues = neededViolinValues() @@ -689,8 +686,8 @@ export function violin( selection ) { var numberOfObjects = violinKeys.length - var min = [].concat(...violinKeys.map(function(k, i){return data[k].quartiles[quartileKeys[0]]})) - var max = [].concat(...violinKeys.map(function(k, i){return data[k].quartiles[quartileKeys[quartileKeys.length - 1]]})) + var min = [].concat(...violinKeys.map(function(k, i){return data[k].utils.math.quartiles[quartileKeys[0]]})) + var max = [].concat(...violinKeys.map(function(k, i){return data[k].utils.math.quartiles[quartileKeys[quartileKeys.length - 1]]})) var extent = [Math.min(...min) - domainPadding, Math.max(...max) + domainPadding] // console.log(extent, violinValues, ordered) @@ -698,9 +695,9 @@ export function violin( selection ) { scale.domain(extent).range(horizontalQ ? [0,spaceY] : [0, spaceX]) var space = horizontalQ ? spaceX : spaceY // calculate object size - objectSize = calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ) + objectSize = utils.math.calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ) // calculate spacer size if needed - spacerSize = calculateWidthOfSpacer(ordered, space, objectSize, numberOfObjects, objectSpacer, overflowQ) + spacerSize = utils.math.calculateWidthOfSpacer(ordered, space, objectSize, numberOfObjects, objectSpacer, overflowQ) // make the nested groups var spacerFunction = groupingSpacer() .horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects) @@ -715,7 +712,7 @@ export function violin( selection ) { // for color function var parentIndexArray = [] container.selectAll('g:not(.to-remove).'+objectClass) - .each(function(d, i){if (hasQ(violinKeys, d)){ parentIndexArray.push(Number(d3.select(this).attr('parent-index')))}}) + .each(function(d, i){if (utils.arr.hasQ(violinKeys, d)){ parentIndexArray.push(Number(d3.select(this).attr('parent-index')))}}) // update color function colorFunction = colorFunction.colorBy() == 'index' @@ -746,20 +743,20 @@ export function violin( selection ) { var t = d3.select(this), currentData = data[key] // needed because bug in selecting .to-remove - if (!hasQ(violinKeys, key)) {return} + if (!utils.arr.hasQ(violinKeys, key)) {return} var i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'), fillColor = colorFunction(key, currentData, i, 'fill'), // prevent duplicate computation strokeColor = colorFunction(key, currentData, i, 'stroke'), - area = safeSelect(t, 'g', 'area'), - la = safeSelect(area, 'path', 'left'), - ra = safeSelect(area, 'path', 'right'), - quarts = safeSelect(t, 'g', 'quarts'), - lq3 = safeSelect(quarts, 'line', 'q3'), - lq1 = safeSelect(quarts, 'line', 'q1'), - q3 = currentData.quartiles[quartileKeys[3]], - q2 = currentData.quartiles[quartileKeys[2]], - q1 = currentData.quartiles[quartileKeys[1]] + area = utils.sel.safeSelect(t, 'g', 'area'), + la = utils.sel.safeSelect(area, 'path', 'left'), + ra = utils.sel.safeSelect(area, 'path', 'right'), + quarts = utils.sel.safeSelect(t, 'g', 'quarts'), + lq3 = utils.sel.safeSelect(quarts, 'line', 'q3'), + lq1 = utils.sel.safeSelect(quarts, 'line', 'q1'), + q3 = currentData.utils.math.quartiles[quartileKeys[3]], + q2 = currentData.utils.math.quartiles[quartileKeys[2]], + q1 = currentData.utils.math.quartiles[quartileKeys[1]] t.attr('transform', horizontalQ ? 'translate('+objectSize / 2+',0)' : 'translate(0,'+objectSize / 2+')' ) // draw curve @@ -786,7 +783,7 @@ export function violin( selection ) { }) if (pointsQ) { - var ptsContainer = safeSelect(t, 'g', 'points') + var ptsContainer = utils.sel.safeSelect(t, 'g', 'points') var pts = ptsContainer.selectAll('.point').data(currentData.pointKeys) pts.on('mouseover', null) @@ -819,7 +816,7 @@ export function violin( selection ) { .attr('cy', function(pointKey, ii){ var dd = currentData.pointValues[ii] if (horizontalQ) { return scale(extent[1]) - scale(dd) } - var j = whichBin(currentData.binned, dd) + var j = utils.arr.whichBin(currentData.binned, dd) var r = Math.random() var n = vScale(r * currentData.frequencies[j] * 0.5) var k = Math.random() > 0.5 ? n : -n @@ -828,7 +825,7 @@ export function violin( selection ) { .attr('cx', function(pointKey, ii){ var dd = currentData.pointValues[ii] if (horizontalQ) { - var j = whichBin(currentData.binned, dd) + var j = utils.arr.whichBin(currentData.binned, dd) var r = Math.random() var n = vScale(r * currentData.frequencies[j] * 0.5) var k = Math.random() > 0.5 ? n : -n @@ -876,11 +873,11 @@ export function violin( selection ) { tooltip() if (tooltip.values() == undefined) { tooltip.values([ - function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] }, - function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] }, - function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] }, - function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] }, - function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] } + function(currentData, tooltipKey){ return currentData['utils.math.quartiles'][tooltipKey] }, + function(currentData, tooltipKey){ return currentData['utils.math.quartiles'][tooltipKey] }, + function(currentData, tooltipKey){ return currentData['utils.math.quartiles'][tooltipKey] }, + function(currentData, tooltipKey){ return currentData['utils.math.quartiles'][tooltipKey] }, + function(currentData, tooltipKey){ return currentData['utils.math.quartiles'][tooltipKey] } ]) } @@ -900,7 +897,7 @@ export function violin( selection ) { * @returns {function} calculateViolinValues * @namespace neededViolinValues */ -function neededViolinValues() { +export function neededViolinValues() { var /** * Whether or not the orientation of the violins are horizontal @@ -911,7 +908,7 @@ function neededViolinValues() { */ horizontalQ = true, /** - * Keys to be put into the quartiles if they need to be calculated. + * Keys to be put into the utils.math.quartiles if they need to be calculated. * (see {@link violin#quartileKeys}) * @param {Object} [quartileKeys=['Q0', 'Q1', 'Q2', 'Q3', 'Q4']] * @memberof neededViolinValues# @@ -998,7 +995,7 @@ function neededViolinValues() { * * data[violinKey].contour // the points depicting the contour of 1/2 of the violin * - * data[violinKey].quartiles // the quartiles of the points' values + * data[violinKey].utils.math.quartiles // the utils.math.quartiles of the points' values * * data[violinKey].pointKeys // the keys associated with the points * @@ -1017,8 +1014,8 @@ function neededViolinValues() { // the numerical values of those points var violinPointsValues = violinPointsKeys.map(function(pk, i){return violinPointValueExtractor(pk, violinPoints)}) - // quartiles of those points - var pointQuartiles = quartiles(violinPointsValues, quartileKeys) + // utils.math.quartiles of those points + var pointQuartiles = utils.math.quartiles(violinPointsValues, quartileKeys) // binned points var binned = d3.histogram()(violinPointsValues) @@ -1039,7 +1036,7 @@ function neededViolinValues() { violinData.binned = binned; violinData.frequencies = frequencies violinData.contour = violinContourPoints - violinData.quartiles = pointQuartiles + violinData.utils.math.quartiles = pointQuartiles violinData.pointKeys = violinPointsKeys violinData.pointValues = violinPointsValues } diff --git a/src/scripts/modules/color-function.js b/src/modules/color-function.js similarity index 98% rename from src/scripts/modules/color-function.js rename to src/modules/color-function.js index 5bef0c5..77035b7 100644 --- a/src/scripts/modules/color-function.js +++ b/src/modules/color-function.js @@ -1,4 +1,4 @@ -import {modifyHexidecimalColorLuminance} from './helpers'; +import {color} from './utils/index.js' /** * Creates a colorFunction @@ -6,7 +6,7 @@ import {modifyHexidecimalColorLuminance} from './helpers'; * @namespace colorFunction * @returns {function} colorFunction */ -export function colorFunction() { +export default function colorFunction() { var data, @@ -30,7 +30,7 @@ export function colorFunction() { * @memberof colorFunction# * @property */ - modifyOpacity = modifyHexidecimalColorLuminance, + modifyOpacity = color.modifyHexidecimalColorLuminance, /** * How to modify color for stroke * @param {number} [strokeOpacity=0] diff --git a/src/scripts/modules/d3-prototypes.js b/src/modules/d3-prototypes.js similarity index 95% rename from src/scripts/modules/d3-prototypes.js rename to src/modules/d3-prototypes.js index db19bf0..3033b24 100644 --- a/src/scripts/modules/d3-prototypes.js +++ b/src/modules/d3-prototypes.js @@ -1,4 +1,4 @@ -import {getContainingSVG} from "./helpers"; +import utils from './utils/index.js' // import * as d3 from "d3"; /******************************************************************************* ** ** @@ -13,7 +13,7 @@ import {getContainingSVG} from "./helpers"; * @augments d3.selection * @returns {Element} which is the svg tag, not the d3 selection of that tag */ -d3.selection.prototype.thisSVG = function() { return getContainingSVG(this.node()); } +d3.selection.prototype.thisSVG = function() { return utils.sel.getContainingSVG(this.node()); } /** diff --git a/src/scripts/modules/grouping-spacer.js b/src/modules/grouping-spacer.js similarity index 97% rename from src/scripts/modules/grouping-spacer.js rename to src/modules/grouping-spacer.js index acb6c6f..e5f808e 100644 --- a/src/scripts/modules/grouping-spacer.js +++ b/src/modules/grouping-spacer.js @@ -1,4 +1,4 @@ -import {log, warn, error, info} from './utils'; +import {con} from './utils/index.js' /******************************************************************************* ** ** ** ** @@ -12,7 +12,7 @@ import {log, warn, error, info} from './utils'; * (see {@link groupingSpacer#recursivelyPosition}) * @namespace groupingSpacer */ -export function groupingSpacer() { +export default function groupingSpacer() { var /*@var {boolean} horizontalQ @default*/ @@ -125,7 +125,7 @@ export function groupingSpacer() { x = horizontalQ ? window.outerWidth : 0, y = !horizontalQ ? window.outerWidth : 0, t = 'translate('+x+','+y+')' - // if(y == undefined) {console.log(cur.node(), y, d)} + // if(y == undefined) {console.con.log(cur.node(), y, d)} return t }) }, @@ -147,7 +147,7 @@ export function groupingSpacer() { * }).remove() */ exitFunction = function(cur){ - log("groupingSpacer", "exiting with", {current: cur, currentNode: cur.node()}) + con.log("groupingSpacer", "exiting with", {current: cur, currentNode: cur.node()}) cur.selectAll('g').classed('to-remove', true) cur.transition().duration(transitionDuration*0.9).ease(easeFunc) @@ -158,7 +158,7 @@ export function groupingSpacer() { x = horizontalQ ? window.outerWidth : 0, y = !horizontalQ ? window.outerWidth : 0, t = 'translate('+x+','+y+')' - // if(y == undefined) {console.log(cur.node(), y, d)} + // if(y == undefined) {console.con.log(cur.node(), y, d)} return t }).remove() } diff --git a/src/scripts/modules/categorical-legend.js b/src/modules/legends/categorical-legend.js similarity index 93% rename from src/scripts/modules/categorical-legend.js rename to src/modules/legends/categorical-legend.js index f20fa7c..7257d9f 100644 --- a/src/scripts/modules/categorical-legend.js +++ b/src/modules/legends/categorical-legend.js @@ -1,10 +1,9 @@ -import {hypenate, safeSelect, round, interpolateColors} from './helpers'; -import {setupContainer, calculateWidthOfObject, calculateWidthOfSpacer} from './utils'; -import {colorFunction as CF} from './color-function'; -import {groupingSpacer} from './grouping-spacer'; -import {unique, flatten} from './array-functions'; +import utils from '../utils/index.js' +import CF from '../color-function'; +import groupingSpacer from '../grouping-spacer'; -export function categoricLegend( selection ) { + +export default function categoricLegend( selection ) { var categories, @@ -351,7 +350,7 @@ export function categoricLegend( selection ) { var verticalQ = !horizontalQ // background cliping rectangle var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY} - var container = setupContainer( selection, namespace, bgcpRect, backgroundFill ); + var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); colorFunction.dataExtent([0, categories.length - 1]) @@ -364,13 +363,13 @@ export function categoricLegend( selection ) { // if grouping is undefined sort barKeys by sortingFunction var ordered = (grouping == undefined) ? categories.sort(sortingFunction) : grouping // ordered might be nested depending on grouping - var catKeys = flatten(ordered) + var catKeys = utils.arr.flatten(ordered) var space = horizontalQ ? spaceX : spaceY // calculate object size - var objectSize = calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ) + var objectSize = utils.math.calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ) // calculate spacer size if needed - var spacerSize = calculateWidthOfSpacer(catKeys, space, objectSize, numberOfObjects, objectSpacer, overflowQ) + var spacerSize = utils.math.calculateWidthOfSpacer(catKeys, space, objectSize, numberOfObjects, objectSpacer, overflowQ) // make the nested groups var spacerFunction = groupingSpacer() .horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects) @@ -383,7 +382,7 @@ export function categoricLegend( selection ) { container.selectAll('g:not(.to-remove).'+objectClass).each(function(cat, i) { var t = d3.select(this) - var c = safeSelect(t, 'circle') + var c = utils.sel.safeSelect(t, 'circle') var fillColor = colorFunction(undefined, cat, i, 'fill'), // prevent duplicate computation strokeColor = colorFunction(undefined, cat, i, 'stroke') @@ -401,7 +400,7 @@ export function categoricLegend( selection ) { .attr('stroke', strokeColor) .attr('stroke-width', bubbleStrokeWidth) - var text = safeSelect(t, 'text') + var text = utils.sel.safeSelect(t, 'text') text.text(cat) .attr('text-anchor', 'middle') .attr("transform", function(d, i){ diff --git a/src/modules/legends/index.js b/src/modules/legends/index.js new file mode 100644 index 0000000..6e8846d --- /dev/null +++ b/src/modules/legends/index.js @@ -0,0 +1,9 @@ +import categorical from './categorical-legend' +import numeric from './numeric-legend' + +let legends = { + categorical, numeric +} + +export {categorical, numeric} +export default legends diff --git a/src/scripts/modules/numeric-legend.js b/src/modules/legends/numeric-legend.js similarity index 75% rename from src/scripts/modules/numeric-legend.js rename to src/modules/legends/numeric-legend.js index 6db2517..bd4824e 100644 --- a/src/scripts/modules/numeric-legend.js +++ b/src/modules/legends/numeric-legend.js @@ -1,9 +1,7 @@ -import {hypenate, safeSelect, round, interpolateColors} from './helpers'; -import {setupContainer} from './utils'; -import {colorFunction as CF} from './color-function'; +import utils from '../utils/index.js' +import CF from '../color-function'; - -export function numericLegend( selection ) { +export default function numericLegend( selection ) { var min=0, @@ -32,15 +30,15 @@ export function numericLegend( selection ) { function legend() { // background cliping rectangle var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY} - var container = setupContainer( selection, namespace, bgcpRect, backgroundFill ); + var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); - var defs = safeSelect(selection, 'defs') - var linearGradient = safeSelect(defs, 'linearGradient') + var defs = utils.sel.safeSelect(selection, 'defs') + var linearGradient = utils.sel.safeSelect(defs, 'linearGradient') .attr("x1", "0%") .attr("y1", "100%") .attr("x2", "0%") .attr("y2", "0%") - .attr('id', hypenate(namespace,'numerical-legend-gradient')) + .attr('id', utils.str.hypenate(namespace,'numerical-legend-gradient')) colorFunction.dataExtent([min, max]) @@ -58,18 +56,18 @@ export function numericLegend( selection ) { - var rect = safeSelect(container, 'rect', 'legend') + var rect = utils.sel.safeSelect(container, 'rect', 'legend') .attr('transform', 'translate(0,'+fontSize+')') - .style("fill", "url(#"+hypenate(namespace,'numerical-legend-gradient')+")") + .style("fill", "url(#"+utils.str.hypenate(namespace,'numerical-legend-gradient')+")") .attr('x', 0) .attr('y', 0) .attr('width', spaceX) .attr('height', spaceY - fontSize*2) .on('mousemove', function(d, i){legendMousemove(d, i, rect, d3.select(this))}) - .on('mouseout', function(d, i){ d3.select("#"+hypenate(namespace,'legend-tooltip')).remove() }) + .on('mouseout', function(d, i){ d3.select("#"+utils.str.hypenate(namespace,'legend-tooltip')).remove() }) - var minText = safeSelect(container, 'text', 'min') - .text(round(min, 2)) + var minText = utils.sel.safeSelect(container, 'text', 'min') + .text(utils.math.round(min, 2)) .attr('text-anchor', 'middle') .attr("font-size", fontSize+'px') .attr('transform', function(d, i){ @@ -80,8 +78,8 @@ export function numericLegend( selection ) { return t }) - var maxText = safeSelect(container, 'text', 'max') - .text(round(max, 2)) + var maxText = utils.sel.safeSelect(container, 'text', 'max') + .text(utils.math.round(max, 2)) .attr('text-anchor', 'middle') .attr("font-size", fontSize+'px') .attr('transform', function(d, i){ @@ -102,13 +100,13 @@ export function numericLegend( selection ) { .domain([0, rect.attr('height')]) .range([max, min]) var m = d3.mouse(rect.node()) - var v = round(s(m[1]),roundTo) + var v = utils.math.round(s(m[1]),roundTo) var strokeColor = colorFunction(undefined, v, undefined, 'stroke') var fillColor = colorFunction(undefined, v, undefined, 'fill') - var div = safeSelect(d3.select('body'), 'div', hypenate(namespace,'legend-tooltip')) - .attr('id', hypenate(namespace,'legend-tooltip')) + var div = utils.sel.safeSelect(d3.select('body'), 'div', utils.str.hypenate(namespace,'legend-tooltip')) + .attr('id', utils.str.hypenate(namespace,'legend-tooltip')) .style('position', 'absolute') .style('left', (d3.event.pageX+15)+'px') .style('top', (d3.event.pageY+15)+'px') @@ -128,7 +126,7 @@ export function numericLegend( selection ) { .style('border-style', 'solid') .style('border-width', 2) - var text = safeSelect(div, 'div') + var text = utils.sel.safeSelect(div, 'div') .text(v) .style('color', textColor) .style('align-self', 'center') diff --git a/src/scripts/modules/tooltip.js b/src/modules/tooltip.js similarity index 85% rename from src/scripts/modules/tooltip.js rename to src/modules/tooltip.js index 77262d5..2ed0abd 100644 --- a/src/scripts/modules/tooltip.js +++ b/src/modules/tooltip.js @@ -1,5 +1,4 @@ -import {safeSelect, round} from './helpers'; -import {log, warn, info, error, consoleGroup, consoleGroupEnd} from './utils'; +import {sel, con, math} from './utils/index.js' /******************************************************************************* ** ** ** ** @@ -15,7 +14,7 @@ import {log, warn, info, error, consoleGroup, consoleGroupEnd} from './utils'; * @returns {tooltip} * @namespace tooltip */ -export function tooltip( selection ) { +export default function tooltip( selection ) { var keys, @@ -83,16 +82,16 @@ export function tooltip( selection ) { * @private */ function mousemove(key, i) { - consoleGroup('d3sm-tooltip') + con.consoleGroup('d3sm-tooltip') var currentData = data[key] var [x, y] = d3.mouse(d3.select("html").node()) - log('tooltip', 'mousemove detected',{key: key, index: i, x:x, y:y}) - log('tooltip', 'current data', currentData) + con.log('tooltip', 'mousemove detected',{key: key, index: i, x:x, y:y}) + con.log('tooltip', 'current data', currentData) - var div = safeSelect(d3.select('html'), 'tooltip', 'd3sm-tooltip') + var div = sel.safeSelect(d3.select('html'), 'tooltip', 'd3sm-tooltip') .classed('card', true) .style('max-width', '300px') .style('background-color', "#212529") @@ -100,14 +99,14 @@ export function tooltip( selection ) { - var cardBody = safeSelect(div, 'div', 'card-body') - var cardTitle = safeSelect(cardBody, 'h5', 'card-title') + var cardBody = sel.safeSelect(div, 'div', 'card-body') + var cardTitle = sel.safeSelect(cardBody, 'h5', 'card-title') .text(header == undefined ? key : typeof header == 'function' ? header(key, currentData, i) : header) .style('color', 'cyan') - var table = safeSelect(cardBody, 'table', 'table').classed('table-dark', true) - var tBody = safeSelect(table, 'tbody') + var table = sel.safeSelect(cardBody, 'table', 'table').classed('table-dark', true) + var tBody = sel.safeSelect(table, 'tbody') tBody = tBody.selectAll('tr') tBody= tBody.data(keys == undefined ? d3.keys(currentData): keys) @@ -120,20 +119,20 @@ export function tooltip( selection ) { .attr('tooltip-row-index', function(d, i){return i}) // tBody = tBody.merge(tr) - consoleGroup('tooltip-rows') + con.consoleGroup('tooltip-rows') tBody.selectAll('.tooltip-key').text(function(d, i){return d}) tBody.selectAll('tr .tooltip-value') .text(function(d, i){ - log('tooltip', 'trying to set value', {rowKey: d, rowIndex: i}) + con.log('tooltip', 'trying to set value', {rowKey: d, rowIndex: i}) var i = d3.select(this).attr('tooltip-row-index') var v = currentData[d]; if (values != undefined) {v = values[i]; if(typeof v == "function") {v = v(currentData, d)}} - return typeof v == 'number' ? round(v, 5) : v + return typeof v == 'number' ? math.round(v, 5) : v }) - consoleGroupEnd() - consoleGroupEnd() + con.consoleGroupEnd() + con.consoleGroupEnd() x += 15 // x += 15 diff --git a/src/scripts/modules/array-functions.js b/src/modules/utils/array.js similarity index 92% rename from src/scripts/modules/array-functions.js rename to src/modules/utils/array.js index 83cb781..fc66bcc 100644 --- a/src/scripts/modules/array-functions.js +++ b/src/modules/utils/array.js @@ -1,11 +1,12 @@ -import {uniqueElements} from './helpers'; -/******************************************************************************* -** ** -** ** -** PROTOTYPES ** -** ** -** ** -*******************************************************************************/ +/** +* Helper function for Array.filter to get unique elements of the array +* @param {*} value current value as mapping over array (self) +* @param {number} index current index in the array +* @param {Array} self passed array from Array.filter method +* @returns {boolean} whether or not value is the first of its kind (i.e. indexOf(value) == index) +*/ +export function uniqueElements(value, index, self) { return self.indexOf(value) === index; } + /** * This function tests to see if all elements of the passed array are true. * @param {Array} array of values diff --git a/src/modules/utils/colors.js b/src/modules/utils/colors.js new file mode 100644 index 0000000..2fb0faf --- /dev/null +++ b/src/modules/utils/colors.js @@ -0,0 +1,32 @@ +/** +* Modifies luminance of hexidecimal number +* @param {string} hex should be hexidecimal value with or without the proceeding octotrope +* @param {number} lum value to increase or decrease luminosity by +* @returns {string} updated hexidecimal value without the proceeding octotrope +*/ +export function modifyHexidecimalColorLuminance(hex, lum) { + // validate hex string + var hex = String(hex).replace(/[^0-9a-f]/gi, ''); + + if (hex.length < 6) { + hex = hex[0]+hex[0]+hex[1]+hex[1]+hex[2]+hex[2]; + } + lum = lum || 0; + + // convert to decimal and change luminosity + var rgb = '#', c, i; + for (i = 0; i < 3; i++) { + c = parseInt(hex.substr(i*2,2), 16); + c = Math.round(Math.min(Math.max(0, c + (c * lum)), 255)).toString(16); + rgb += ('00'+c).substr(c.length); + } + + return rgb; +} + +/** +* Maps arguments in to d3.interpolateRgbBasis +* @param arguments +* @returns {Function} +*/ +export function interpolateColors(){return d3.interpolateRgbBasis(arguments)} diff --git a/src/modules/utils/console.js b/src/modules/utils/console.js new file mode 100644 index 0000000..8da9cfa --- /dev/null +++ b/src/modules/utils/console.js @@ -0,0 +1,97 @@ +/** + * calls console.group if d3sm.debugQ == true + * @param {string} name of the group + * @returns {undefined} + */ +export function consoleGroup(name) { + if (window.d3sm.debugQ === true){ + console.group(name) + } +} + +/** + * calls console.groupEnd if d3sm.debugQ == true + * @returns {undefined} + */ +export function consoleGroupEnd() { + if (window.d3sm.debugQ === true){ + console.groupEnd() + } +} + +/** + * Calls console.log if d3sm.debugQ == true + * @param {string} func name of the function logging + * @param {string} msg to log + * @param {Object} data to be logged along side the message + * @returns {undefined} + */ +export function log(func, msg, data) { + if (window.d3sm.debugQ === true){ + console.log( + `%c[d3sm::${func}]:\t${msg}`, + [ + 'background: #6cd1ef', + 'border-radius: 5000px', + 'padding: 0px 2px', + 'font-size: 14px' + ].join(';') + ) + console.table(data) + // console.trace() + } +} + +/** + * Calls console.warn if d3sm.debugQ == true + * @param {string} func name of the function warning + * @param {string} msg to display + * @param {Object} data to be displayed along side the message + * @returns {undefined} + */ +export function warn(func, msg, data) { + if (window.d3sm.debugQ === true) + console.warn( + `%c[d3sm::${func}]:\t${msg}`, + [ + 'background: #ffd53e', + 'border-radius: 5000px', + 'padding: 0px 2px', + 'font-size: 14px' + ].join(';') + ) + console.table(data) +} +/** + * Calls the console.info if d3sm.debugQ == true + * @param {string} func name of the function providing info + * @param {string} msg to display + * @param {Object} data to be displayed along side the message + * @returns {undefined} + */ +export function info(func, msg, data) { + if (window.d3sm.debugQ) + console.info( + `%c[d3sm::${func}]:\t${msg}`, + [ + 'background: #009ccd', + 'border-radius: 5000px', + 'padding: 0px 2px', + 'font-size: 14px' + ].join(';') + ) + console.table(data) +} + + +/** + * Calls console.error if d3sm.debugQ == true + * @param {string} func name of the function which sends the error + * @param {string} msg to display + * @param {Object} data to be displayed along side the message + * @returns {undefined} + */ +export function error(func, msg, data) { + if (window.d3sm.debugQ) + console.error(`[d3sm::${func}]:\t${msg}\t%o`,data) +} diff --git a/src/modules/utils/index.js b/src/modules/utils/index.js new file mode 100644 index 0000000..e117b00 --- /dev/null +++ b/src/modules/utils/index.js @@ -0,0 +1,16 @@ +import * as arr from './array.js' +import * as color from './colors.js' +import * as math from './math.js' +import * as misc from './misc.js' +import * as sel from './selections.js' +import * as str from './strings.js' +import * as con from './console.js' +import * as paths from './paths.js' + + +let utils = { + arr, color, math, misc, sel, str, con, paths +} + +export {arr, color, math, misc, sel, str, con, paths} +export default utils diff --git a/src/modules/utils/math.js b/src/modules/utils/math.js new file mode 100644 index 0000000..eac6234 --- /dev/null +++ b/src/modules/utils/math.js @@ -0,0 +1,160 @@ +/** +* Rounds decimals of number to precision +* @param {number} number +* @param {number} precision +* @returns {number} rounded to precision +*/ +export function round(number, precision) { + var shift = function (number, precision, reverseShift) { + if (reverseShift) { + precision = -precision; + } + var numArray = ('' + number).split('e'); + return +(numArray[0] + 'e' + (numArray[1] ? (+numArray[1] + precision) : precision)); + }; + return shift(Math.round(shift(number, precision, false)), precision, true); +} + +/** +* evenly partitions the range [min, max] into n parts +* @param {number} min +* @param {number} max +* @param {number} n +* @returns {number[]} array of length n evenly partitioned between min and max +*/ +export function tickRange(min, max, n) { + var a = [min] + var d = max-min + var s = d / (n-1) + for (var i = 0; i < n-2; i++) { a.push(min + s * (i+1)) } + a.push(max) + return a +} + +/** +* @deprecated @see{@link tickRange} +* @param {number} min +* @param {number} max +* @param {number} parts +* @returns {number[]} array of length parts evenly partitioned between min and max +*/ +export function partitionRangeInto(min, max, parts) { + var diff = max - min + return Array(parts).map(function (e, i) { return min + diff / parts * i }) +} + + +export function euclideanDistance(p1, p2){ + var a = p1[0] - p2[0], b = p1[1] - p2[1] + return Math.sqrt(a*a + b*b) +} + +/** +* Calculated the quartiles of the passed data and stores them with qKeys +* @param {number[]} data list of numerical values +* @param {string[]} [qKeys=['q0', 'q1', 'q2', 'q3', 'q4']] how returned object with quartiles should be stored +* @returns {Object} with keys qKeys giving only the numerical values for the quartiles +*/ +export function quartiles(data, qKeys) { + var + q2 = d3.median(data), + lower = data.filter(x => x < q2), + upper = data.filter(x => x > q2), + + q1 = d3.median(lower), + q1 = q1 == undefined ? q2 : q1, + + q0 = d3.min(lower), + q0 = q0 == undefined ? q1 : q0, + + q3 = d3.median(upper), + q3 = q3 == undefined ? q2 : q3, + + q4 = d3.max(upper), + q4 = q4 == undefined ? q3 : q4, + + k0 = 'q0', k1 = 'q1', k2 = 'q2', k3 = 'q3', k4 = 'q4', + obj = {} + if (qKeys!=undefined && qKeys.length == 5) { k0 = qKeys[0]; k1 = qKeys[1]; k2 = qKeys[2]; k3 = qKeys[3]; k4 = qKeys[4]; } + obj[k0] = q0; obj[k1] = q1; obj[k2] = q2; obj[k3] = q3; obj[k4] = q4; + + return obj +} + + +/** +* determines the width of an object for the calling plotting function +* @param {number} freeSpace how much space is avalible +* @param {number} numberOfObjects how many object do we need +* @param {number} minObjectWidth how small are these objects allowed to be +* @param {number} maxObjectWidth how large are these object allowed to be +* @param {number} sizeOfSpacer percent of freeSpace that a single spacer should take up (need numberOfObjects - 1 spacers) +* @param {boolean} overflowQ can we go beyond alloted space +* @returns {number} how large object should be +* function tries to keep object within min / max width, but wil default to +* 5e-10 (smallest consistenly visible by svg size of element) if overflowQ is false +*/ +export function calculateWidthOfObject(freeSpace, numberOfObjects, minObjectWidth, maxObjectWidth, sizeOfSpacer, overflowQ) { + var sizeOfSpacer = + sizeOfSpacer == 0 || sizeOfSpacer > 1 + ? sizeOfSpacer + : freeSpace * sizeOfSpacer + + var numberOfSpacers = numberOfObjects - 1 + var spaceTakenBySpacers = numberOfSpacers * sizeOfSpacer + var remainingSpace = freeSpace - spaceTakenBySpacers + remainingSpace = remainingSpace < 0 ? 0 : remainingSpace + var objectWidth = remainingSpace / numberOfObjects + + if ( overflowQ && minObjectWidth != undefined && objectWidth < minObjectWidth ) { objectWidth = minObjectWidth } + // if ( maxObjectWidth != undefined && objectWidth > maxObjectWidth ) { objectWidth = maxObjectWidth } + if ( overflowQ && maxObjectWidth != undefined && objectWidth < maxObjectWidth ) { objectWidth = maxObjectWidth } + return Math.max(objectWidth, 5e-10) +} + +/** +* @param {Array[]} data list data (can be nested). If nested will create more complex spacer size +* @param {number} freeSpace how much space is avalible +* @param {number} objectWidth @see{@link calculateWidthOfObject} +* @param {number} numberOfObjects how many object do we need +* @param {number} baseSpacerSize percent of freeSpace that a single spacer should take up (need numberOfObjects - 1 spacers) +* @param {boolean} overflowQ can we go beyond alloted space +* @returns {number} returns size that spacer should be at level=0 +*/ +export function calculateWidthOfSpacer(data, freeSpace, objectWidth, numberOfObjects, baseSpacerSize, overflowQ) { + if (overflowQ) { + // var limitedNumberOfObjects = numberOfObjects > 6 ? 6 : numberOfObjects + // var spaceLeft = freeSpace - limitedNumberOfObjects * objectWidth + // return spaceLeft / (limitedNumberOfObjects - 1) + return freeSpace * baseSpacerSize + } + var spacersAtEachLevel = spacersNeededAtEachLevel(data) + var totalSpacerPercent = total(spacersAtEachLevel.map(function(e, i) {return e * 1 / (i+1)})) + var baseSpacerSize = (freeSpace - (objectWidth * numberOfObjects)) / totalSpacerPercent + // console.log(freeSpace, objectWidth, numberOfObjects, totalSpacerPercent) + // console.log(totalSpacerPercent, baseSpacerSize, totalSpacerPercent * baseSpacerSize) + return isNaN(baseSpacerSize) ? 0 : baseSpacerSize +} + +/** +* Calculates number of spacers needed to seperate elements at each level. +* @param {Array[]} array list data (can be nested). If nested will create more complex spacer size +* @param {number} [level=0] current level, used in recusrion +* @param {Array} [levelData=[]] how many spacers needed at a given level +* @returns {Array} levelData +* +* @example +* array = [[1,2], [3,4]] +* // returns [1, 2] +* as at level=0 the only spacer needed is between [1,2] and [3,4] +* and at level=1 the only two spacers needed is between 1 and 2 as well as +* 3 and 4 since the spacer between 2 and 3 is handled at level=0 +*/ +export function spacersNeededAtEachLevel (array, level, levelData ) { + if ( level == undefined ) { level = 0; } else { level += 1 } + if ( levelData == undefined ) { levelData = []; } + if ( level >= levelData.length ) { levelData.push(array.length - 1) } + else { levelData[level] += array.length - 1 } + array.map(function(e, i) { if (Array.isArray(e)) { spacersNeededAtEachLevel(e, level, levelData) } } ) + return levelData +} diff --git a/src/modules/utils/misc.js b/src/modules/utils/misc.js new file mode 100644 index 0000000..ac8344f --- /dev/null +++ b/src/modules/utils/misc.js @@ -0,0 +1,139 @@ + +export function resizeDebounce(f, wait) { + var resize = debounce(function(){f()},wait) + window.addEventListener('resize', resize) +} + +function debounce(func, wait, immediate) { + var timeout; + return function() { + var context = this, args = arguments; + var later = function() { + timeout = null; + if (!immediate) func.apply(context, args); + }; + var callNow = immediate && !timeout; + clearTimeout(timeout); + timeout = setTimeout(later, wait); + if (callNow) func.apply(context, args); + }; +} + +export function setupStandardChartContainers( + selection, + namespace, + container, + margins={top:0.01, bottom:0.01, left:0.01, right:0.01}, + svg={w:0.8, h:0.6}, // percent of container space for svg + axes={y:0.1, x:0.1}, // percent of container space for axes, + leg={x:0, margin:0, pos:'left'} // absolute width of legend and space on either size +) +{ + if (container == undefined) {container = {w:window.innerWidth, h:window.Height}} + // SVG width and height + + var svgSpace = { + w: container.w * svg.w, + h: container.h * svg.h + } + + var margPx = { + top: margins.top * svgSpace.h, + bottom: margins.bottom * svgSpace.h, + left: margins.left * svgSpace.w, + right: margins.right * svgSpace.w + }, + + + + // Space after removing margins + chartSpace = { + w: svgSpace.w - margPx.left - margPx.right, + h: svgSpace.h - margPx.top - margPx.bottom + }, + + // main dimension of x and y axies + // e.g. defines how tall x axis is as length is determined by plotRect.w + axesSpace = { + x: chartSpace.h * axes.x, + y: chartSpace.w * axes.y + }, + + // space left for drawing the chart properly (e.g. bars, violins, etc) + drawingSpace = { + x: chartSpace.w - axesSpace.y - leg.x - 2*leg.margin, + y: chartSpace.h - axesSpace.x + }, + + + legRect = { + x: leg.margin + margPx.left + (leg.pos == 'left' ? 0 : drawingSpace.x + axesSpace.y), + y: margPx.top, // this is soomehow getting calculated incorectly + w: leg.x, + h: drawingSpace.y + }, + + yAxisRect = { + x: axesSpace.y + margPx.left + (leg.pos == 'left' ? leg.x + 2*leg.margin : 0), + y: margPx.top, + w: axesSpace.y, + h: drawingSpace.y + }, + + plotRect = { + x: axesSpace.y + margPx.left + (leg.pos == 'left' ? leg.x + 2*leg.margin : 0), + y: margPx.top, + w: drawingSpace.x, + h: drawingSpace.y + }, + + xAxisRect = { + x: axesSpace.y + margPx.left + (leg.pos == 'left' ? leg.x + 2*leg.margin : 0), + y: margPx.top + drawingSpace.y, + w: drawingSpace.x, + h: axesSpace.x + } + + + + container = d3sm.safeSelect(selection, 'svg', namespace) + .style('width', svgSpace.w+'px') + .style('height', svgSpace.h+'px') + + var axes = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'axes')) + + var leg = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'legend')) + .attr('transform', "translate("+legRect.x+","+legRect.y+")") + + var plot = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'plot')) + .attr('transform', "translate("+plotRect.x+","+plotRect.y+")") + + var xAxis = d3sm.safeSelect(axes, 'g', d3sm.hypenate(namespace, 'x-axis')) + .attr('transform', "translate("+xAxisRect.x+","+xAxisRect.y+")") + + var yAxis = d3sm.safeSelect(axes, 'g', d3sm.hypenate(namespace, 'y-axis')) + .attr('transform', "translate("+yAxisRect.x+","+yAxisRect.y+")") + + return { + svg: { + selection: container, + rect: svgSpace + }, + plot: { + selection: plot, + rect: plotRect + }, + xAxis: { + selection: xAxis, + rect: xAxisRect + }, + yAxis: { + selection: yAxis, + rect: yAxisRect + }, + legend: { + selection: leg, + rect: legRect + } + } +} diff --git a/src/modules/utils/paths.js b/src/modules/utils/paths.js new file mode 100644 index 0000000..cafc8dc --- /dev/null +++ b/src/modules/utils/paths.js @@ -0,0 +1,40 @@ + +/** +* Draws a whisker for @see{@link boxwhisker} +* @param {boolean} dir direction to draw whisker, should be either true (up, top) or false (down or bottom) +* @param {number} x starting x coordinate in which to draw whisker +* @param {number} y starting y coordinate in which to draw whisker +* @param {number} w width of space in which to draw whisker +* @param {number} h height of space in which to draw whisker +* @param {number} per percentage of w or h (depends on o) to make whisker +* @param {boolean} o orientation, true is horizontal and false is vertical +* @returns {string} representing the svg path (i.e. the d attribute for a path tag) +*/ +export function whiskerPath(dir, x, y, w, h, per, o) { + // d = direction (true is up), p = percent width + if (dir == 'up' || dir == 'top' || dir == true) {dir = true} + if (dir == 'down' || dir == 'bottom' || dir == false) {dir = false} + o = o == undefined ? 'horizontal' : o + per = per == undefined ? 1 : per + if (o != "horizontal") { + var hh = h * per , + w = dir ? w : -w , + a = dir ? x + w : x , + b = dir ? x : x + w , + c = dir ? a : b + p = "M " + a + ' ' + ( h / 2 ) + ' ' + + 'L ' + b + ' ' + ( h / 2 ) + ' ' + + 'M ' + c + ' ' + ( h / 2 - hh / 2 ) + ' ' + + 'L ' + c + ' ' + ( h / 2 + hh / 2 ) + ' ' + + return p + } + var ww = w * per, + a = dir ? y + h : y , + b = dir ? y : y + h , + p = "M " + ( w / 2 ) + ' ' + a + ' ' // straight line part + + 'L ' + ( w / 2 ) + ' ' + b + ' ' // straight line part + + 'h ' + ( -ww / 2 ) + ' ' + 0 + ' ' // horizontal line part + + 'h ' + ( ww ) + ' ' + 0 + ' ' + return p +} diff --git a/src/modules/utils/selections.js b/src/modules/utils/selections.js new file mode 100644 index 0000000..044855a --- /dev/null +++ b/src/modules/utils/selections.js @@ -0,0 +1,120 @@ +import {hypenate} from './strings.js' +/** +* Extracts x and y of translate from transform property +* @param {string} transform transform property of svg element +* @returns {number[]} x, y of translate(x, y) +*/ +export function getTranslation(transform) { + // Create a dummy g for calculation purposes only. This will never + // be appended to the DOM and will be discarded once this function + // returns. + var g = document.createElementNS('http://www.w3.org/2000/svg', 'g'); + // Set the transform attribute to the provided string value. + transform = transform == undefined ? 'translate(0,0)' : transform; + g.setAttributeNS(null, 'transform', transform); + // consolidate the SVGTransformList containing all transformations + // to a single SVGTransform of type SVG_TRANSFORM_MATRIX and get + // its SVGMatrix. + var matrix = g.transform.baseVal.consolidate().matrix; + // As per definition values e and f are the ones for the translation. + return [matrix.e, matrix.f]; +} + +/** +* recursively ascends element.parentElement to find a svg tag +* @param {Element} element +* @returns {Element | undefined} +*/ +export function getContainingSVG(element) { + var parent = element.parentElement + var tag = parent.tagName.toLowerCase() + if (tag === 'svg') { return parent; } + if (tag === 'html') { return undefined; } + return getContainingSVG(parent); +} + + +/** +* Trys to use d3.selection to get element, if it doesnt exist, makes one +* @param {d3.selection} sel selection in which to try and find object +* @param {string} tag tag of which to try and select +* @param {string} [cls=''] class of tag to try and grab +* @returns {d3.selection} of either append or selected tag.cls within sel +*/ +export function safeSelect(sel, tag, cls) { + var clsStr = cls == undefined ? '' : '.'+cls; + var sSel = sel.select(tag+clsStr).empty() + ? sel.append(tag) + : sel.select(tag+clsStr) + return sSel + .classed(clsStr.replace('.', ''), true) + .attr('transform', sSel.attr('transform') == undefined ? 'translate(0,0)' : sSel.attr('transform')) +} + + + +/** +* Adds a clip-path rect and binds it to container +* @param {d3.selection} container in which to add the clip-path and to which to bind the cliping path to +* @param {Object} rect the coordinates (x, y, width, height) of the clip-path +* @param {string} namespace +* @returns {d3.selection} of the clip-path rect +*/ +export function cpRect(container, rect, namespace) { + var defs = safeSelect(container, 'defs', hypenate(namespace, 'definitions')) + var cp = safeSelect(defs, 'clipPath', hypenate(namespace, 'clip-path')) + .attr('id', hypenate(namespace, 'clip-path')) + + var cpRect = safeSelect(cp, 'rect') + .attr('x', rect.x) + .attr('y', rect.y) + .attr('width', rect.width) + .attr('height', rect.height) + + defs.raise() + // set clipping path to container + container.attr('clip-path', 'url(#'+ hypenate(namespace, 'clip-path')+')') + + return cpRect +} + + +/** +* Adds a background rect t to container +* @param {d3.selection} container in which to add the background rectangle +* @param {Object} rect the coordinates (x, y, width, height) of the background +* @param {string} fill the color of the background +* @returns {d3.selection} of the background fill +*/ +export function bgRect(container, rect, fill) { + return safeSelect(container, 'rect', 'bg') + .attr('x', rect.x) + .attr('y', rect.y) + .attr('width', rect.width) + .attr('height', rect.height) + .attr('fill', fill) +} + + +/** +* Sets up the container for making chart elements. This includes making +* a clip-path rect bound to the passed container, a background rect, and +* a g element with class -object-container. +* @param {d3.selection} container in which to add the clip-path and background +* @param {string} namespace +* @param {Object} rect the coordinates (x, y, width, height) of the background and clip-path +* @param {string} fill the color of the background +* @returns {d3.selection} of g.-object-container +* +* @see{@link bgRect} +* @see{@link cpRect} +*/ +export function setupContainer(selection, namespace, rect, fill) { + // the container for three main items, bg, defs, and object-container + var + container = safeSelect(selection, 'g', namespace), + bg = bgRect(container, rect, fill), + cp = cpRect(container, rect, namespace), + objectContainer = safeSelect(container, 'g', hypenate(namespace, 'object-container')) + return objectContainer +} diff --git a/src/modules/utils/strings.js b/src/modules/utils/strings.js new file mode 100644 index 0000000..e1ae5c5 --- /dev/null +++ b/src/modules/utils/strings.js @@ -0,0 +1,41 @@ + +/** +* Hypenates all strings together +* @param {string[]} arguments +* @returns {string} "arg1-arg2-...-argn" +*/ +export function hypenate(){ return Array.prototype.slice.call(arguments).join('-') } + + +/** +* Trys to reduce text to fit in specified area, made for tick labels as called by +* @see{@link axis} +* @param {d3.selection} t container for specific axis tick +* @param {string} text to be the label of the passed axis tick +* @param {boolean} orient of the axis, true is horizontal, false is vertical +* @param {number} tickLength is the length of the text +* @param {number} space is the amount of availble space for the text and the tick to fit in +* @param {boolean} overflowQ whether or not allowed to go over the alloted space +* @returns {none} +*/ +export function truncateText(t, text, orient, tickLength, space, overflowQ) { + var rect = t.node().getBoundingClientRect() + t.text(text) + while (Math.max(rect.width, rect.height) > space - tickLength) { + text = String(text) + text = text.slice(0, text.length - 1) + t.text(text + '...') + rect = t.node().getBoundingClientRect() + if (text.length == 0) break + } +} + + +export function truncateString(string, space, font) { + var chars = space / font; + if (chars < string.length) { + return string.slice(0, Math.round(chars-5)) + '...' + } else { + return string + } +} diff --git a/src/scripts/main.js b/src/scripts/main.js deleted file mode 100644 index 737feaf..0000000 --- a/src/scripts/main.js +++ /dev/null @@ -1,116 +0,0 @@ -// Import styles (automatically inject into ). -// import '../styles/main.css'; -import {axis} from './modules/axis'; -import {bar} from './modules/bar'; -import {bubbleHeatmap} from './modules/bubble-heatmap'; -import {heatmap} from './modules/heatmap'; -import {boxwhisker} from './modules/box-whisker'; -import {colorFunction} from './modules/color-function'; -import {datatoggle} from './modules/data-toggle'; -import {groupingSpacer} from './modules/grouping-spacer'; -import {tooltip} from './modules/tooltip'; -import {scatter} from './modules/scatter'; -import {plotZoom} from './modules/plot-zoom'; -import {multiPlotZoom} from './modules/multi-plot-zoom'; -import {violin} from './modules/violin'; -import {numericLegend} from './modules/numeric-legend'; -import {categoricLegend} from './modules/categorical-legend'; -import {lasso} from './modules/lasso'; -import {lassoWidget} from './modules/lasso-widget'; -import {selectFilter} from './modules/select-filter'; -import {upset} from './modules/upset'; -import {filterTable} from './modules/filter-table'; - -import {uniqueElements, getTranslation, modifyHexidecimalColorLuminance, tickRange, -quartiles, extractViolinValues, hypenate, round, getContainingSVG, -interpolateColors, truncateText, safeSelect} from './modules/helpers'; - -import { - all, tally, hasQ, first, last, total, unique, get, listOfListsQ, - cut, groupBy, arrayEquals, elementsAtLevels, numberOfElements, - flatten, whichBin -} from './modules/array-functions'; - - -import { - setupStandardChartContainers, log as myLog, warn, info, error, - consoleGroup, consoleGroupEnd, resizeDebounce -} from './modules/utils'; - -// /** @module d3sm */ -var d3sm = {}; -d3sm.axis = axis; -d3sm.bar = bar; -d3sm.bubbleHeatmap = bubbleHeatmap; -d3sm.heatmap = heatmap; -d3sm.boxwhisker = boxwhisker; -d3sm.colorFunction = colorFunction; -d3sm.datatoggle = datatoggle; -d3sm.groupingSpacer = groupingSpacer; -d3sm.tooltip = tooltip; -d3sm.scatter = scatter; -d3sm.plotZoom = plotZoom; -d3sm.multiPlotZoom = multiPlotZoom; -d3sm.violin = violin; -d3sm.numericLegend = numericLegend; -d3sm.categoricLegend = categoricLegend; -d3sm.lasso = lasso; -d3sm.lassoWidget = lassoWidget; -d3sm.selectFilter = selectFilter; -d3sm.upset = upset; -d3sm.filterTable = filterTable; - -d3sm.uniqueElements = uniqueElements; -d3sm.getTranslation = getTranslation; -d3sm.modifyHexidecimalColorLuminance = modifyHexidecimalColorLuminance; -d3sm.tickRange = tickRange; -d3sm.quartiles = quartiles; -d3sm.extractViolinValues = extractViolinValues; -d3sm.hypenate = hypenate; -d3sm.round = round; -d3sm.getContainingSVG = getContainingSVG; -d3sm.interpolateColors = interpolateColors; -d3sm.truncateText = truncateText; -d3sm.safeSelect = safeSelect; - -d3sm.whichBin = whichBin; -d3sm.unique = unique; -d3sm.flatten = flatten; - -d3sm.setupStandardChartContainers = setupStandardChartContainers; -d3sm.log = myLog; -d3sm.warn = warn; -d3sm.info = info; -d3sm.error = error; -d3sm.consoleGroup = consoleGroup; -d3sm.consoleGroupEnd = consoleGroupEnd; -d3sm.resizeDebounce = resizeDebounce; - -d3sm.debugQ = false - - - -// Import a logger for easier debugging -// import debug from 'debug'; -// const log = debug('app:log'); - -// The logger should only be disabled if we're not in production. -// if (ENV !== 'production') { -// // Enable the logger. -// debug.enable('*'); -// log('Logging is enabled!'); -// -// // Enable LiveReload -// document.write( -// ' + + + + + + + + + + + + + + + + + + +
+ + + +
+

By Value

+
+ +
+

By Category

+
+ +
+

By Value Guidelines

+
+
+ + + + + diff --git a/demos/axes/js/v001.js b/demos/axes/js/v001.js new file mode 100644 index 0000000..50cacce --- /dev/null +++ b/demos/axes/js/v001.js @@ -0,0 +1,209 @@ +var data = d3smDemoData['basicViolins'] + +var +selection = d3.select("#axes-by-value") +namespace = 'axes', +container = d3sm.utils.sel.safeSelect(selection, 'svg', namespace).style('width', '1000px'), +lAxis = d3sm.utils.sel.safeSelect(container, 'g', d3sm.utils.str.hypenate(namespace, 'left-axis')).attr('transform', "translate(100,100)"), +rAxis = d3sm.utils.sel.safeSelect(container, 'g', d3sm.utils.str.hypenate(namespace, 'right-axis')).attr('transform', "translate(900,100)") +uAxis = d3sm.utils.sel.safeSelect(container, 'g', d3sm.utils.str.hypenate(namespace, 'up-axis')).attr('transform', "translate(100,100)") +dAxis = d3sm.utils.sel.safeSelect(container, 'g', d3sm.utils.str.hypenate(namespace, 'down-axis')).attr('transform', "translate(100,400)") + + +tickValues = [1,2,3,4,5] + +var uA = d3sm.axis(uAxis) +.spaceX(800) +.spaceY(100) +.orient("top") +.overflowQ(false) +.tickValues(tickValues) +.numberOfTicks(5) +.namespace('u') +// .guideLinesQ(true) +.guidelineSpace(300) + +var dA = d3sm.axis(dAxis) +.spaceX(800) +.spaceY(100) +.orient("bottom") +.label('bottom') +.overflowQ(false) +.tickValues(tickValues) +.numberOfTicks(5) +.namespace('d') +// .guideLinesQ(true) +.guidelineSpace(300) +.tickLabelRotation(315) +// .tickTickLabelSpacer(0) + + + +var lA = d3sm.axis(lAxis) +.spaceX(100) +.spaceY(300) +.orient("left") +.label('left') +.overflowQ(false) +.tickValues(tickValues) +.numberOfTicks(5) +.namespace('l') +// .guideLinesQ(true) +.guidelineSpace(800) +// .tickLabelRotation(45) + + +var rA = d3sm.axis(rAxis) +.spaceX(100) +.spaceY(300) +.orient("right") +.overflowQ(false) +.tickValues(tickValues) +.numberOfTicks(5) +.namespace('r') +// .guideLinesQ(true) +.guidelineSpace(800) +// .tickLabelRotation(45) + + +uA() +dA() +lA() +rA() + + + +var +selection = d3.select("#axes-by-cat") +namespace = 'axes', +container = d3sm.utils.sel.safeSelect(selection, 'svg', namespace).style('width', '1000px'), +lAxis = d3sm.utils.sel.safeSelect(container, 'g', d3sm.utils.str.hypenate(namespace, 'left-axis')).attr('transform', "translate(100,100)"), +rAxis = d3sm.utils.sel.safeSelect(container, 'g', d3sm.utils.str.hypenate(namespace, 'right-axis')).attr('transform', "translate(900,100)") +uAxis = d3sm.utils.sel.safeSelect(container, 'g', d3sm.utils.str.hypenate(namespace, 'up-axis')).attr('transform', "translate(100,100)") +dAxis = d3sm.utils.sel.safeSelect(container, 'g', d3sm.utils.str.hypenate(namespace, 'down-axis')).attr('transform', "translate(100,400)") + + +tickValues = [1,2,3,4,5] +tickLabels = ["this", "this-text", "this-text-gets", "this-text-gets-longer", "this-text-gets-longer-as-we-go-on"] + +var uA = d3sm.axis(uAxis) +.spaceX(800) +.spaceY(100) +.label('top') +.orient("top") +.overflowQ(false) +.categoricalQ(true) +.tickLabels(tickLabels) +.numberOfTicks(5) +.namespace('u-c') +// .guideLinesQ(true) +.guidelineSpace(300) + +var dA = d3sm.axis(dAxis) +.spaceX(800) +.spaceY(100) +.orient("bottom") +.overflowQ(false) +.categoricalQ(true) +.tickLabels(tickLabels) +.numberOfTicks(5) +.namespace('d-c') +// .guideLinesQ(true) +.guidelineSpace(300) +// .tickLabelRotation(315) +.tickTickLabelSpacer(0) + + +var lA = d3sm.axis(lAxis) +.spaceX(100) +.spaceY(300) +.orient("left") +.overflowQ(false) +.categoricalQ(true) +.tickLabels(tickLabels) +.numberOfTicks(5) +.namespace('l-c') +// .guideLinesQ(true) +.guidelineSpace(800) + +var rA = d3sm.axis(rAxis) +.spaceX(100) +.spaceY(300) +.label('right') +.orient("right") +.overflowQ(false) +.categoricalQ(true) +.tickLabels(tickLabels) +.numberOfTicks(5) +.namespace('r-c') +// .guideLinesQ(true) +.guidelineSpace(800) + +uA() +dA() +lA() +rA() + + + +var +selection = d3.select("#axes-by-val-g") +namespace = 'axes', +container = d3sm.utils.sel.safeSelect(selection, 'svg', namespace).style('width', '1000px'), +lAxis = d3sm.utils.sel.safeSelect(container, 'g', d3sm.utils.str.hypenate(namespace, 'left-axis')).attr('transform', "translate(100,100)"), +rAxis = d3sm.utils.sel.safeSelect(container, 'g', d3sm.utils.str.hypenate(namespace, 'right-axis')).attr('transform', "translate(900,100)") +uAxis = d3sm.utils.sel.safeSelect(container, 'g', d3sm.utils.str.hypenate(namespace, 'up-axis')).attr('transform', "translate(100,100)") +dAxis = d3sm.utils.sel.safeSelect(container, 'g', d3sm.utils.str.hypenate(namespace, 'down-axis')).attr('transform', "translate(100,400)") + + +tickValues = [1,2,3,4,5] + +var uA = d3sm.axis(uAxis) +.spaceX(800) +.spaceY(100) +.orient("top") +.overflowQ(false) +.tickValues(tickValues) +.numberOfTicks(5) +.namespace('u-g') +.guideLinesQ(true) +.guidelineSpace(300) + +var dA = d3sm.axis(dAxis) +.spaceX(800) +.spaceY(100) +.orient("bottom") +.overflowQ(false) +.tickValues(tickValues) +.numberOfTicks(5) +.namespace('d-g') +.guideLinesQ(true) +.guidelineSpace(300) + + +var lA = d3sm.axis(lAxis) +.spaceX(100) +.spaceY(300) +.orient("left") +.overflowQ(false) +.tickValues(tickValues) +.numberOfTicks(5) +.namespace('l-g') +.guideLinesQ(true) +.guidelineSpace(800) + +var rA = d3sm.axis(rAxis) +.spaceX(100) +.spaceY(300) +.orient("right") +.overflowQ(false) +.tickValues(tickValues) +.numberOfTicks(5) +.namespace('r-g') +.guideLinesQ(true) +.guidelineSpace(800) + +uA() +dA() +lA() +rA() diff --git a/demos/bar-chart-same-data-complex-grouping/index.html b/demos/bar-chart-same-data-complex-grouping/index.html new file mode 100644 index 0000000..cfdb93c --- /dev/null +++ b/demos/bar-chart-same-data-complex-grouping/index.html @@ -0,0 +1,215 @@ + + + + + Complex Grouping + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/demos/bar-chart-same-data-complex-grouping/js/v001.js b/demos/bar-chart-same-data-complex-grouping/js/v001.js new file mode 100644 index 0000000..0d79b66 --- /dev/null +++ b/demos/bar-chart-same-data-complex-grouping/js/v001.js @@ -0,0 +1,125 @@ +function barPlot( selection ) { + var data, + namespace + plot.data = function(_){return arguments.length ? (data = _, plot) : data; }; + plot.namespace = function(_){return arguments.length ? (namespace = _, plot) : namespace; }; + + + function plot() { + var + databar = d3sm.d3sm.utils.sel.safeSelect(selection, 'form',d3sm.utils.str.hypenate(namespace,'databar')), + dataPrint = d3sm.d3sm.utils.sel.safeSelect(selection, 'div', d3sm.utils.str.hypenate(namespace,'data-print')), + + selections = d3sm.setupStandardChartContainers(selection, namespace, {w: 1000, h:500}, undefined, {axes:{x:0.1, y:0.1},space:{w:1,h:1}}) + container = selections.svg.selection, + bars = selections.plot.selection, + xAxis = selections.xAxis.selection, + yAxis = selections.yAxis.selection + + + // make data selection + var dataSelect = d3sm.datatoggle(databar) + .keys(d3.keys(data.groupings)) + .updateFunction(function(){plot(); zoom.reset()}) + .namespace(namespace)() + + // grab current selected key + var currentKey = dataSelect.currentKey() + // // extract subdata + // var currentData = data[currentKey] + // + + + dataPrint.text(JSON.stringify(data.groupings[currentKey])) + // make bars + var b = d3sm.bar(bars) + .data(data.data) + .grouping(data.groupings[currentKey]) + .spaceX(selections.plot.rect.w) + .spaceY(selections.plot.rect.h) + .overflowQ(true) + .orient('horizontal') + .maxObjectSize(20) + .minObjectSize(10) + .valueExtractor(function(d, i){return data.data[d]['value'] }) + + + // b.tooltip().keys(['value']) + // + // b.colorFunction().colorBy('index') + b() + + // extract keys and values to safe re-computation + var bk = b.barKeys(), bv = b.barValues() + + // left d3sm.axis + var yA = d3sm.axis(yAxis) + .spaceX(selections.yAxis.rect.w) + .spaceY(selections.yAxis.rect.h) + .orient("left") + .overflowQ(false) + .tickValues(bv) + .guideLinesQ(true) + .guidelineSpace(selections.xAxis.rect.w) + .namespace('axis-left') + .numberOfTicks(5) + yA() + + // bottom d3sm.axis + var xA = d3sm.axis(xAxis) + .spaceX(selections.xAxis.rect.w) + .spaceY(selections.xAxis.rect.h) + .orient("bottom") + .overflowQ(true) + .categoricalQ(true) + .tickLabels(bk) + .namespace('axis-bottom') + .minObjectSize(b.minObjectSize()) + .maxObjectSize(b.maxObjectSize()) + .grouping(data.groupings[currentKey]) + .objectSpacer(b.objectSpacer()) + xA() + + b.selection().selectAll("rect").nodes().map(function(d, i){ + console.log(d.getBoundingClientRect(), i) + return this + }), + console.log( + b.selection().node(), + b.selection().selectAll("rect").nodes(), + b.selection().selectAll("rect").nodes(), + 'BCR:',b.selection().node().getBoundingClientRect(), + 'BBox:',b.selection().node().getBBox() + ) + + + + + + // // set-up zoom, whell and drag + var zoom = d3sm.plotZoom(b, xA, yA).eventType('drag') + .orient('horizontal') + // .xLock( + // objs.getBBox().width + // ) + + console.log(zoom) + + // zoom.setLocks() + + var pan = d3.zoom().scaleExtent([1,1]).on('zoom', function(){ + // zoom.setLocks() + zoom.eventType('drag')() + }) + container.call(pan).on('wheel', function(){ + // zoom.setLocks() + + zoom.eventType('wheel')() + }) + } + + + + + return plot +} diff --git a/demos/basic-violins/index.html b/demos/basic-violins/index.html new file mode 100644 index 0000000..6dd08db --- /dev/null +++ b/demos/basic-violins/index.html @@ -0,0 +1,182 @@ + + + + + Basic Violins + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/demos/box-whiskers/index.html b/demos/box-whiskers/index.html new file mode 100644 index 0000000..ab45578 --- /dev/null +++ b/demos/box-whiskers/index.html @@ -0,0 +1,171 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/demos/bubble-heatmap/index.html b/demos/bubble-heatmap/index.html new file mode 100644 index 0000000..423b32b --- /dev/null +++ b/demos/bubble-heatmap/index.html @@ -0,0 +1,169 @@ + + + + + Basic Bubble Heatmap + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/demos/bubble-heatmap/js/v000.js b/demos/bubble-heatmap/js/v000.js new file mode 100644 index 0000000..03adb85 --- /dev/null +++ b/demos/bubble-heatmap/js/v000.js @@ -0,0 +1,96 @@ +function heatmap( selection ) { + var data, + namespace + plot.data = function(_){return arguments.length ? (data = _, plot) : data; }; + plot.namespace = function(_){return arguments.length ? (namespace = _, plot) : namespace; }; + + function plot(){ + var + selections = d3sm.setupStandardChartContainers(selection, namespace, {w: 1000, h:750}, undefined, {axes:{x:0.2, y:0.2},space:{w:1,h:1}}) + container = selections.svg.selection, + bubbles = selections.plot.selection, + xAxis = selections.xAxis.selection, + yAxis = selections.yAxis.selection + + // d3sm.debugQ = true + + + var bhm = d3sm.bubbleHeatmap(bubbles) + .spaceX(selections.plot.rect.w) + .spaceY(selections.plot.rect.h) + .xKey('xKey') + .yKey('yKey') + .vKey('val') + .rKey('val') + .overflowQ(true) + // .minObjectSize(100) + .maxObjectSize(100) + + + // .yKeySortingFunction() + // .xKeySortingFunction(function(a,b){return data[a]['xKey']}) + + .data(data) + + bhm.tooltip().keys(['tooltip']) + bhm() + // + // + // + // // extract keys and values to safe re-computation + var xk = bhm.xValues(), yk = bhm.yValues() + // var pd = [] + // vk.map(function(k, i){ pd.push(...d3.values(vv[k].quartiles)) }) + // pd = d3sm.unique(pd.sort(function (a, b) {return d3.ascending(a, b)})) + // + // console.log(pd) + // console.log(yk, bhm.minObjectSize(), bhm.maxObjectSize(), bhm.ySpacerSize()) + + var yA = d3sm.axis(yAxis) + .spaceX(selections.yAxis.rect.w) + .spaceY(selections.yAxis.rect.h) + .orient("left") + .overflowQ(true) + .categoricalQ(true) + .tickLabels(yk) + .namespace('axis-left') + .minObjectSize(bhm.minObjectSize()) + .maxObjectSize(bhm.maxObjectSize()) + .objectSpacer(bhm.ySpacerSize()) + .objectSize(bhm.ySize()) + + .guideLinesQ(true) + .guidelineSpace(selections.xAxis.rect.w) + yA() + + var xA = d3sm.axis(xAxis) + .spaceX(selections.xAxis.rect.w) + .spaceY(selections.xAxis.rect.h) + .orient("bottom") + .overflowQ(true) + .categoricalQ(true) + .tickLabels(xk) + .namespace('axis-bottom') + .minObjectSize(bhm.minObjectSize()) + .maxObjectSize(bhm.maxObjectSize()) + .objectSpacer(bhm.xSpacerSize()) + .objectSize(bhm.xSize()) + .guideLinesQ(true) + .guidelineSpace(selections.yAxis.rect.h) + + xA() + + console.log(bhm.selection().node(), bhm.selection().node().getBBox(), bhm.selection().node().getBoundingClientRect()) + + + var zoom = d3sm.plotZoom(bhm, xA, yA).eventType('drag') + .orient('2D') + + var pan = d3.zoom().scaleExtent([1,1]).on('zoom', function(){zoom.eventType('drag')()}) + container.call(pan).on('wheel', function(){ zoom.eventType('wheel')() }) + + } + + + return plot +} diff --git a/demos/heatmap/index.html b/demos/heatmap/index.html new file mode 100644 index 0000000..504290f --- /dev/null +++ b/demos/heatmap/index.html @@ -0,0 +1,70 @@ + + + + + Basic Heatmap + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/demos/heatmap/js/v000.js b/demos/heatmap/js/v000.js new file mode 100644 index 0000000..2dff8e0 --- /dev/null +++ b/demos/heatmap/js/v000.js @@ -0,0 +1,98 @@ +function heatmap( selection ) { + var data, + namespace + plot.data = function(_){return arguments.length ? (data = _, plot) : data; }; + plot.namespace = function(_){return arguments.length ? (namespace = _, plot) : namespace; }; + + function plot(){ + var + selections = d3sm.setupStandardChartContainers(selection, namespace, {w: 1000, h:750}, undefined, {axes:{x:0.2, y:0.2},space:{w:1,h:1}}) + container = selections.svg.selection, + bubbles = selections.plot.selection, + xAxis = selections.xAxis.selection, + yAxis = selections.yAxis.selection + + + d3sm.debugQ = true + + var overflow = true + // var overflow = false + var bhm = d3sm.heatmap(bubbles) + .spaceX(selections.plot.rect.w) + .spaceY(selections.plot.rect.h) + .xKey('xKey') + .yKey('yKey') + .vKey('val') + .overflowQ(overflow) + .objectStrokeWidth(2) + // .minObjectSize(100) + .maxObjectSize(100) + + + // .yKeySortingFunction() + // .xKeySortingFunction(function(a,b){return data[a]['xKey']}) + + .data(data) + + bhm.tooltip().keys(['tooltip']) + bhm() + // + // + // + // // extract keys and values to safe re-computation + var xk = bhm.xValues(), yk = bhm.yValues() + // var pd = [] + // vk.map(function(k, i){ pd.push(...d3.values(vv[k].quartiles)) }) + // pd = d3sm.unique(pd.sort(function (a, b) {return d3.ascending(a, b)})) + // + // console.log(pd) + // console.log(yk, bhm.minObjectSize(), bhm.maxObjectSize(), bhm.ySpacerSize()) + + var yA = d3sm.axis(yAxis) + .spaceX(selections.yAxis.rect.w) + .spaceY(selections.yAxis.rect.h) + .orient("left") + .overflowQ(overflow) + .categoricalQ(true) + .tickLabels(yk) + .namespace('axis-left') + .minObjectSize(bhm.minObjectSize()) + .maxObjectSize(bhm.maxObjectSize()) + .objectSpacer(bhm.ySpacerSize()) + .objectSize(bhm.ySize()) + + .guideLinesQ(true) + .guidelineSpace(selections.xAxis.rect.w) + yA() + + var xA = d3sm.axis(xAxis) + .spaceX(selections.xAxis.rect.w) + .spaceY(selections.xAxis.rect.h) + .orient("bottom") + .overflowQ(overflow) + .categoricalQ(true) + .tickLabels(xk) + .namespace('axis-bottom') + .minObjectSize(bhm.minObjectSize()) + .maxObjectSize(bhm.maxObjectSize()) + .objectSpacer(bhm.xSpacerSize()) + .objectSize(bhm.xSize()) + .guideLinesQ(true) + .guidelineSpace(selections.yAxis.rect.h) + + xA() + + console.log(bhm.selection().node(), bhm.selection().node().getBBox(), bhm.selection().node().getBoundingClientRect()) + + + var zoom = d3sm.plotZoom(bhm, xA, yA).eventType('drag') + .orient('2D') + + var pan = d3.zoom().scaleExtent([1,1]).on('zoom', function(){zoom.eventType('drag')()}) + container.call(pan).on('wheel', function(){ zoom.eventType('wheel')() }) + + } + + + return plot +} diff --git a/demos/heatmap/js/v003.js b/demos/heatmap/js/v003.js new file mode 100644 index 0000000..1846d59 --- /dev/null +++ b/demos/heatmap/js/v003.js @@ -0,0 +1,132 @@ +function heatmap ( selection ) { + var + chart, + xAxis, + yAxis, + legend, + + data, + namespace = 'heatmap', + title = 'Heatmap' + + plot.data = function(_){return arguments.length ? (data = _, plot) : data; }; + plot.title = function(_){return arguments.length ? (title = _, plot) : title; }; + plot.namespace = function(_){return arguments.length ? (namespace = _, plot) : namespace; }; + + plot.chart = function(_){return arguments.length ? (chart = _, plot) : chart; }; + plot.xAxis = function(_){return arguments.length ? (xAxis = _, plot) : xAxis; }; + plot.yAxis = function(_){return arguments.length ? (yAxis = _, plot) : yAxis; }; + plot.legend = function(_){return arguments.length ? (legend = _, plot) : legend; }; + + function plot(){ + var + orient = '2D', + + titleContainer = d3sm.utils.sel.safeSelect(selection, 'h5', d3sm.utils.str.hypenate(namespace,'title')).text(title), + + selections = d3sm.utils.misc.setupStandardChartContainers( + selection, + namespace, + {w:Math.min(window.innerWidth,1200), h:Math.min(window.innerHeight,700)}, + {top:0.01, bottom:0.01, left:0.01, right:0.01}, + {w:1, h:1}, // percent of container space for svg + {y:0.1,x:0.2}, // percent of container space for axes, + {x:25, margin:05} // absolute width of legendSelect and space on either size + ), + + container = selections.svg.selection, + chartSelection = selections.plot.selection, + xAxisSelection = selections.xAxis.selection, + yAxisSelection = selections.yAxis.selection, + legendSelect = selections.legend.selection + + + if (chart == undefined){ chart = d3sm.charts.heatmap(chartSelection) } + if (yAxis == undefined) { yAxis = d3sm.axis(yAxisSelection) } + if (xAxis == undefined) { xAxis = d3sm.axis(xAxisSelection) } + if (legend == undefined) { + legend = d3sm.legends.numeric(legendSelect) + .spaceX(selections.legend.rect.w).spaceY(selections.plot.rect.h) + + } + + // d3sm.debugQ = true + var overflowQ = true + + chart + .spaceX(selections.plot.rect.w) + .spaceY(selections.plot.rect.h) + .xKey('ykey') + .yKey('xkey') + .vKey('padj') + // .minObjectSize(25) + // .maxObjectSize(100) + // .domainPadding(0.01) + .overflowQ(overflowQ) + .data(data) + + chart.tooltip() + .header(function(k, cD){return cD.experimentId}) + .keys(['padj', 'log2FoldChange', 'xkey', 'ykey']) + + chart.colorFunction().colorBy('value') + .valueExtractor(function(k, v, i){ return data[k].padj }) + + chart() + + legend + .min(d3.min(chart.vValues())) + .max(d3.max(chart.vValues())) + .colorFunction(chart.colorFunction()) + .roundTo(5) + legend() + + var xk = chart.xValues(), yk = chart.yValues() + + // console.log(xk, yk) + yAxis + .spaceX(selections.yAxis.rect.w) + .spaceY(selections.yAxis.rect.h) + .orient("left") + .overflowQ(overflowQ) + .categoricalQ(true) + .tickLabels(yk) + .namespace(d3sm.utils.str.hypenate(namespace,'axis-left')) + .minObjectSize(chart.yMinObjectSize()) + .maxObjectSize(chart.yMaxObjectSize()) + .objectSpacer(chart.objectSpacer()) + .spacerSize(0) + .objectSize(chart.ySize() + chart.ySpacerSize()) + .guideLinesQ(false) + .guidelineSpace(selections.xAxis.rect.w) + + yAxis() + + xAxis = d3sm.axis(xAxisSelection) + .spaceX(selections.xAxis.rect.w) + .spaceY(selections.xAxis.rect.h) + .orient("bottom") + .overflowQ(overflowQ) + .categoricalQ(true) + .tickLabels(xk) + .namespace(d3sm.utils.str.hypenate(namespace,'axis-bottom')) + .minObjectSize(chart.xMinObjectSize()) + .maxObjectSize(chart.xMaxObjectSize()) + .objectSpacer(chart.objectSpacer()) + .spacerSize(0) + .objectSize(chart.xSize() + chart.xSpacerSize()) + .guideLinesQ(false) + .guidelineSpace(selections.yAxis.rect.h) + xAxis() + + var zoom = d3sm.aux.plotZoom(chart, xAxis, yAxis).eventType('drag') + .orient("vertical") + + var pan = d3.zoom().scaleExtent([1,1]).on('zoom', function(){zoom.eventType('drag')()}) + chartSelection.call(pan).on('wheel', function(){ zoom.eventType('wheel')() }) + + } + + +return plot +} diff --git a/demos/index.html b/demos/index.html index 92a645f..3f6112a 100644 --- a/demos/index.html +++ b/demos/index.html @@ -8,32 +8,43 @@ - - - - - - diff --git a/demos/scatter/index.html b/demos/scatter/index.html new file mode 100644 index 0000000..15c2ce7 --- /dev/null +++ b/demos/scatter/index.html @@ -0,0 +1,144 @@ + + + + + Scatter Plot + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + +
+
+
+ + +
+
Rendered with v0.0.3
+ + +
+ + + + + + + + + diff --git a/demos/scatter/js/v001.js b/demos/scatter/js/v001.js new file mode 100644 index 0000000..6d0bea9 --- /dev/null +++ b/demos/scatter/js/v001.js @@ -0,0 +1,103 @@ +//VERSION: 0.0.1 +function scatterPlot( selection ) { + var data, + namespace + plot.data = function(_){return arguments.length ? (data = _, plot) : data; }; + plot.namespace = function(_){return arguments.length ? (namespace = _, plot) : namespace; }; + + + function plot() { + var + databar = d3sm.safeSelect(selection, 'form', d3sm.hypenate(namespace,'databar')), + + selections = d3sm.setupStandardChartContainers(selection, namespace, {w: 1000, h:500}, undefined, {axes:{x:0.1, y:0.1},space:{w:1,h:1}}) + container = selections.svg.selection, + chart = selections.plot.selection, + xAxis = selections.xAxis.selection, + yAxis = selections.yAxis.selection + + leg = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace,'legend')) + .attr('transform','translate(10, 24)') + + var dataSelect = d3sm.datatoggle(databar) + .keys(d3.keys(data.groupings)) + .namespace(namespace) + .updateFunction(function(){ plot(); zoom.reset(); }) + () + + var currentKey = dataSelect.currentKey() + + var currentData = d3.keys(data.data).reduce( (acc, k) => { + if (data.groupings[currentKey].includes(k)) { acc[k] = data.data[k] } + return acc + }, {}) + + + var scale = currentKey == "linearScale" ? d3.scaleLinear() : d3.scaleSqrt() + var ticks = currentKey == "linearScale" ? 5 : 10 + + var s = d3sm.scatter(chart) + .data(currentData) + .spaceX(selections.plot.rect.w) + .spaceY(selections.plot.rect.h) + .domainPaddingX(0.1) + .domainPaddingY(0.1) + .valueExtractorR(function(d, i){return currentData[d]['r']}) + .maxRadius(20) + .minRadius(1) + .scaleY(scale) + .scaleR(d3.scalePow()) + + var rs = d3.keys(currentData).map(function(d, i){return currentData[d].r}) + + s.colorFunction().colorBy('value').valueExtractor(function(k, v, i){ return v.r }) + .dataExtent( [d3.min(rs), d3.max(rs)]) + + s() + + var xV = s.valuesX(), yV = s.valuesY() + + var yA = d3sm.axis(yAxis) + .spaceX(selections.yAxis.rect.w) + .spaceY(selections.yAxis.rect.h) + .orient("left") + .overflowQ(false) + .tickValues(yV) + .guideLinesQ(true) + .guidelineSpace(selections.xAxis.rect.w) + .namespace('axis-left') + .numberOfTicks(ticks) + .domainPadding(s.domainPaddingY()) + .scale(scale) + yA() + + var xA = d3sm.axis(xAxis) + .spaceX(selections.xAxis.rect.w) + .spaceY(selections.xAxis.rect.h) + .orient("bottom") + .overflowQ(false) + .tickValues(xV) + .guideLinesQ(true) + .guidelineSpace(selections.yAxis.rect.h) + .namespace('axis-bottom') + .numberOfTicks(ticks) + .domainPadding(s.domainPaddingX()) + + xA() + + // set-up zoom, whell and drag + var zoom = d3sm.plotZoom(s, xA, yA).eventType('drag').orient('2D') + var pan = d3.zoom().scaleExtent([1,1]).on('zoom', function(){ zoom.eventType('drag')() }) + container.call(pan).on('wheel', function(){ zoom.eventType('wheel')() }) + + l = d3sm.numericLegend(leg) + .min(d3.min(s.valuesR())) + .max(d3.max(s.valuesR())) + .spaceX(25) + .spaceY(300) + + l() + + } + return plot +} diff --git a/demos/scatter/js/v002.js b/demos/scatter/js/v002.js new file mode 100644 index 0000000..eb27c94 --- /dev/null +++ b/demos/scatter/js/v002.js @@ -0,0 +1,117 @@ +//VERSION: 0.0.2 +function scatterPlot( selection ) { + var data, + namespace + plot.data = function(_){return arguments.length ? (data = _, plot) : data; }; + plot.namespace = function(_){return arguments.length ? (namespace = _, plot) : namespace; }; + + + function plot() { + var + databar = d3sm.safeSelect(selection, 'form', d3sm.hypenate(namespace,'databar')), + + selections = d3sm.setupStandardChartContainers(selection, namespace, {w: 1000, h:500}, undefined, {axes:{x:0.1, y:0.1},space:{w:1,h:1}}) + container = selections.svg.selection, + chart = selections.plot.selection, + xAxis = selections.xAxis.selection, + yAxis = selections.yAxis.selection + + leg = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace,'legend')) + .attr('transform','translate(10, 24)') + + var dataSelect = d3sm.datatoggle(databar) + .keys(d3.keys(data.groupings)) + .namespace(namespace) + .updateFunction(function(){ plot(); zoom.reset(); }) + () + + var currentKey = dataSelect.currentKey() + + var currentData = d3.keys(data.data).reduce( (acc, k) => { + if (data.groupings[currentKey].includes(k)) { acc[k] = data.data[k] } + return acc + }, {}) + + + var scale = currentKey == "linearScale" ? d3.scaleLinear() : d3.scaleSqrt() + var ticks = currentKey == "linearScale" ? 5 : 10 + + var s = d3sm.scatter(chart) + .data(currentData) + .spaceX(selections.plot.rect.w) + .spaceY(selections.plot.rect.h) + .domainPaddingX(0.1) + .domainPaddingY(0.1) + .valueExtractorR(function(d, i){return currentData[d]['r']}) + .maxRadius(20) + .minRadius(1) + .scaleY(scale) + .scaleR(d3.scalePow()) + + var rs = d3.keys(currentData).map(function(d, i){return currentData[d].r}) + + s.colorFunction().colorBy('value').valueExtractor(function(k, v, i){ return v.r }) + .dataExtent( [d3.min(rs), d3.max(rs)]) + + s() + + var xV = s.valuesX(), yV = s.valuesY() + + var yA = d3sm.axis(yAxis) + .spaceX(selections.yAxis.rect.w) + .spaceY(selections.yAxis.rect.h) + .orient("left") + .overflowQ(false) + .tickValues(yV) + .guideLinesQ(true) + .guidelineSpace(selections.xAxis.rect.w) + .namespace('axis-left') + .numberOfTicks(ticks) + .domainPadding(s.domainPaddingY()) + .scale(scale) + yA() + + var xA = d3sm.axis(xAxis) + .spaceX(selections.xAxis.rect.w) + .spaceY(selections.xAxis.rect.h) + .orient("bottom") + .overflowQ(false) + .tickValues(xV) + .guideLinesQ(true) + .guidelineSpace(selections.yAxis.rect.h) + .namespace('axis-bottom') + .numberOfTicks(ticks) + .domainPadding(s.domainPaddingX()) + + xA() + // + // + // set-up zoom, whell and drag + var zoom = d3sm.plotZoom(s, xA, yA).eventType('drag').orient('2D') + var pan = d3.zoom().scaleExtent([1,1]).on('zoom', function(){ zoom.eventType('drag')() }) + container.call(pan).on('wheel', function(){ zoom.eventType('wheel')() }) + + l = d3sm.numericLegend(leg) + .min(d3.min(s.valuesR())) + .max(d3.max(s.valuesR())) + .spaceX(25) + .spaceY(300) + + l() + + // l = d3sm.categoricLegend(leg) + // .categories(['a','b','c']) + // .spaceX(100) + // .spaceY(300) + // .overflowQ(false) + // .orient('vertical') + // + // l() + + } + + + + + return plot +} diff --git a/demos/scatter/js/v003.js b/demos/scatter/js/v003.js new file mode 100644 index 0000000..b111659 --- /dev/null +++ b/demos/scatter/js/v003.js @@ -0,0 +1,187 @@ +//VERSION: 0.0.3 +function scatterPlot( selection ) { + var + data, + namespace, + chart, + xAxis, + yAxis, + dataSelect, + legend, + lasso, + lassoWidget + + plot.data = function(_){return arguments.length ? (data = _, plot) : data; }; + plot.namespace = function(_){return arguments.length ? (namespace = _, plot) : namespace; }; + plot.chart = function(_){return arguments.length ? (chart = _, plot) : chart; }; + plot.lasso = function(_){return arguments.length ? (lasso = _, plot) : lasso; }; + + function plot() { + var objectClass= 'item' + + var + lassoDiv = d3sm.safeSelect(selection, 'div', d3sm.hypenate(namespace,'lassoWidget')), + databar = d3sm.safeSelect(selection, 'form', d3sm.hypenate(namespace,'databar')), + selections = d3sm.setupStandardChartContainers( + selection, + namespace, + {w:window.innerWidth, h:window.innerHeight}, + {top:0.01, bottom:0.01, left:0.01, right:0.01}, + {w: 0.8, h:0.8}, // percent of container space for svg + {y:0.1,x:0.1}, // percent of container space for axes, + {x:25, margin:05} // absolute width of legendSelect and space on either size + ), + + // selections = d3sm.setupStandardChartContainers(selection, namespace, {w: 1000, h:500}, undefined, {axes:{x:0.1, y:0.1},space:{w:1,h:1}}) + container = selections.svg.selection, + chartSelection = selections.plot.selection, + xAxisSelection = selections.xAxis.selection, + yAxisSelection = selections.yAxis.selection, + legendSelect = selections.legend.selection + + + var + xAxisToggle = { 'linear': d3.scaleLinear(), 'sqrt': d3.scaleSqrt() }, + yAxisToggle = { 'linear': d3.scaleLinear(), 'sqrt': d3.scaleSqrt() }, + rAxisToggle = { 'linear': d3.scaleLinear(), 'sqrt': d3.scaleSqrt(), 'pow': d3.scalePow() } + + + if (dataSelect == undefined) { dataSelect = d3sm.datatoggle(databar) + .data({ + 'dataset': data.groupings, + 'x-axis': xAxisToggle, + 'y-axis': yAxisToggle, + 'radius scale': rAxisToggle + }) + .namespace(namespace) + .updateFunction(function(){ plot(); zoom.reset(); }) + dataSelect() + } + + if (chart == undefined) { chart = d3sm.scatter(chartSelection) } + if (yAxis == undefined) { yAxis = d3sm.axis(yAxisSelection) } + if (xAxis == undefined) { xAxis = d3sm.axis(xAxisSelection) } + if (legend == undefined) {legend = d3sm.numericLegend(legendSelect) } + + + var currentKeys = dataSelect.currentKeys() + var xScale = xAxisToggle[currentKeys['x-axis']] + var yScale = yAxisToggle[currentKeys['y-axis']] + var rScale = rAxisToggle[currentKeys['radius scale']] + var currentDataKey = currentKeys['dataset'] + + + var currentKey = dataSelect.currentKeys() + var currentData = d3.keys(data.data).reduce( (acc, k) => { + if (data.groupings[currentDataKey].includes(k)) { acc[k] = data.data[k] } + return acc + }, {}) + + + chart + .data(currentData) + .spaceX(selections.plot.rect.w) + .spaceY(selections.plot.rect.h) + .domainPaddingX(0.1) + .domainPaddingY(0.1) + .valueExtractorR(function(d, i){return currentData[d]['r']}) + .maxRadius(20) + .minRadius(1) + .scaleY(yScale) + .scaleX(xScale) + .scaleR(rScale) + .namespace(namespace) + .objectClass(objectClass) + + var rs = d3.keys(currentData).map(function(d, i){return currentData[d].r}) + + chart.colorFunction().colorBy('value').valueExtractor(function(k, v, i){ return v.r }) + .dataExtent( [d3.min(rs), d3.max(rs)]) + chart() + + var xV = chart.valuesX(), yV = chart.valuesY() + + yAxis + .spaceX(selections.yAxis.rect.w) + .spaceY(selections.yAxis.rect.h) + .orient("left") + .overflowQ(false) + .tickValues(yV) + .guideLinesQ(true) + .guidelineSpace(selections.xAxis.rect.w) + .namespace('axis-left') + .numberOfTicks(5) + .domainPadding(chart.domainPaddingY()) + .scale(yScale) + yAxis() + + xAxis = d3sm.axis(xAxisSelection) + .spaceX(selections.xAxis.rect.w) + .spaceY(selections.xAxis.rect.h) + .orient("bottom") + .overflowQ(false) + .tickValues(xV) + .guideLinesQ(true) + .guidelineSpace(selections.yAxis.rect.h) + .namespace('axis-bottom') + .numberOfTicks(5) + .domainPadding(chart.domainPaddingX()) + .scale(xScale) + + xAxis() + + + legend + .min(d3.min(chart.valuesR())) + .max(d3.max(chart.valuesR())) + .spaceX(selections.legend.rect.w) + .spaceY(selections.legend.rect.h) + + legend() + + + + // set-up zoom, whell and drag + var zoom = d3sm.plotZoom(chart, xAxis, yAxis).eventType('drag').orient('2D') + var pan = d3.zoom().scaleExtent([1,1]).on('zoom', function(){ zoom.eventType('drag')() }) + chartSelection.call(pan).on('wheel', function(){ zoom.eventType('wheel')() }) + + + // var svg = selection.select('svg') + // var objs = svg.select('g.'+namespace+'-object-container') + // var objC = '.chart-point' + + // if (lasso == undefined) { lasso = d3sm.lasso( ) + // .svg(svg) + // .objectClass(objC) + // .objectContainer(objs) + // .namespace('test') + // .chartContainer(chartSelection) + // } + // + // lasso + // .xScale(chart.scaleX()) + // .yScale(chart.scaleY()) + // + // lasso() + + + if (lassoWidget == undefined) {lassoWidget = d3sm.lassoWidget(lassoDiv) + .svg(selections.svg.selection) + .objectClass("."+objectClass) + .chartContainer(chartSelection) + .objectContainer(chartSelection.select('g.'+namespace+'-object-container')) + .maxNumberOfGroups(2) + .dataExtractor(function(nodeData){ + return {"name": nodeData} + }) + lassoWidget() + } + + } + + + + + return plot +} diff --git a/demos/scatter/js/v004.js b/demos/scatter/js/v004.js new file mode 100644 index 0000000..9f0e5ee --- /dev/null +++ b/demos/scatter/js/v004.js @@ -0,0 +1,194 @@ +//VERSION: 0.0.4 +function scatterPlot( selection ) { + var + data, + namespace, + chart, + xAxis, + yAxis, + dataSelect, + legend, + lasso, + lassoWidget + + plot.data = function(_){return arguments.length ? (data = _, plot) : data; }; + plot.namespace = function(_){return arguments.length ? (namespace = _, plot) : namespace; }; + plot.chart = function(_){return arguments.length ? (chart = _, plot) : chart; }; + plot.lasso = function(_){return arguments.length ? (lasso = _, plot) : lasso; }; + + function plot() { + var objectClass= 'item' + + var + lassoDiv = d3sm.utils.sel.safeSelect(selection, 'div', d3sm.utils.str.hypenate(namespace,'lassoWidget')), + databar = d3sm.utils.sel.safeSelect(selection, 'form', d3sm.utils.str.hypenate(namespace,'databar')), + selections = d3sm.utils.misc.setupStandardChartContainers( + selection, + namespace, + {w:window.innerWidth, h:window.innerHeight}, + {top:0.01, bottom:0.01, left:0.01, right:0.01}, + {w: 0.8, h:0.8}, // percent of container space for svg + {y:0.1,x:0.1}, // percent of container space for axes, + {x:25, margin:05} // absolute width of legendSelect and space on either size + ), + + // selections = d3sm.setupStandardChartContainers(selection, namespace, {w: 1000, h:500}, undefined, {axes:{x:0.1, y:0.1},space:{w:1,h:1}}) + container = selections.svg.selection, + chartSelection = selections.plot.selection, + xAxisSelection = selections.xAxis.selection, + yAxisSelection = selections.yAxis.selection, + legendSelect = selections.legend.selection + + + var + xAxisToggle = { 'linear': d3.scaleLinear(), 'sqrt': d3.scaleSqrt() }, + yAxisToggle = { 'linear': d3.scaleLinear(), 'sqrt': d3.scaleSqrt() }, + rAxisToggle = { 'linear': d3.scaleLinear(), 'sqrt': d3.scaleSqrt(), 'pow': d3.scalePow() } + + // + if (dataSelect == undefined) { dataSelect = d3sm.aux.datatoggle(databar) + .data({ + 'dataset': data.groupings, + 'x-axis': xAxisToggle, + 'y-axis': yAxisToggle, + 'radius scale': rAxisToggle + }) + .namespace(namespace) + .updateFunction(function(){ plot(); zoom.reset(); }) + dataSelect() + } + + if (chart == undefined) { chart = d3sm.charts.scatter(chartSelection) } + if (yAxis == undefined) { yAxis = d3sm.axis(yAxisSelection) } + if (xAxis == undefined) { xAxis = d3sm.axis(xAxisSelection) } + if (legend == undefined) {legend = d3sm.legends.numeric(legendSelect) } + + + var currentKeys = dataSelect.currentKeys() + var xScale = xAxisToggle[currentKeys['x-axis']] + var yScale = yAxisToggle[currentKeys['y-axis']] + var rScale = rAxisToggle[currentKeys['radius scale']] + var currentDataKey = currentKeys['dataset'] + + // + // var xScale = xAxisToggle['linear'] + // var yScale = yAxisToggle['linear'] + // var rScale = rAxisToggle['linear'] + // var currentDataKey = 'datasetOne' + + + + var currentKey = dataSelect.currentKeys() + var currentData = d3.keys(data.data).reduce( (acc, k) => { + if (data.groupings[currentDataKey].includes(k)) { acc[k] = data.data[k] } + return acc + }, {}) + + + chart + .data(currentData) + .spaceX(selections.plot.rect.w) + .spaceY(selections.plot.rect.h) + .domainPaddingX(0.1) + .domainPaddingY(0.1) + .valueExtractorR(function(d, i){return currentData[d]['r']}) + .maxRadius(20) + .minRadius(1) + .scaleY(yScale) + .scaleX(xScale) + .scaleR(rScale) + .namespace(namespace) + .objectClass(objectClass) + + var rs = d3.keys(currentData).map(function(d, i){return currentData[d].r}) + + chart.colorFunction().colorBy('value').valueExtractor(function(k, v, i){ return v.r }) + .dataExtent( [d3.min(rs), d3.max(rs)]) + chart() + + var xV = chart.valuesX(), yV = chart.valuesY() + + yAxis + .spaceX(selections.yAxis.rect.w) + .spaceY(selections.yAxis.rect.h) + .orient("left") + .overflowQ(false) + .tickValues(yV) + .guideLinesQ(true) + .guidelineSpace(selections.xAxis.rect.w) + .namespace('axis-left') + .numberOfTicks(5) + .domainPadding(chart.domainPaddingY()) + .scale(yScale) + yAxis() + + xAxis = d3sm.axis(xAxisSelection) + .spaceX(selections.xAxis.rect.w) + .spaceY(selections.xAxis.rect.h) + .orient("bottom") + .overflowQ(false) + .tickValues(xV) + .guideLinesQ(true) + .guidelineSpace(selections.yAxis.rect.h) + .namespace('axis-bottom') + .numberOfTicks(5) + .domainPadding(chart.domainPaddingX()) + .scale(xScale) + + xAxis() + + + legend + .min(d3.min(chart.valuesR())) + .max(d3.max(chart.valuesR())) + .spaceX(selections.legend.rect.w) + .spaceY(selections.legend.rect.h) + + legend() + + + + // set-up zoom, whell and drag + var zoom = d3sm.aux.plotZoom(chart, xAxis, yAxis).eventType('drag').orient('2D') + var pan = d3.zoom().scaleExtent([1,1]).on('zoom', function(){ zoom.eventType('drag')() }) + chartSelection.call(pan).on('wheel', function(){ zoom.eventType('wheel')() }) + + + var svg = selection.select('svg') + var objs = svg.select('g.'+namespace+'-object-container') + var objC = '.chart-point' + + if (lasso == undefined) { lasso = d3sm.aux.lasso( ) + .svg(svg) + .objectClass(objC) + .objectContainer(objs) + .namespace('test') + .chartContainer(chartSelection) + } + + lasso + .xScale(chart.scaleX()) + .yScale(chart.scaleY()) + + lasso() + + + if (lassoWidget == undefined) {lassoWidget = d3sm.aux.lassoWidget(lassoDiv) + .svg(selections.svg.selection) + .objectClass("."+objectClass) + .chartContainer(chartSelection) + .objectContainer(chartSelection.select('g.'+namespace+'-object-container')) + .maxNumberOfGroups(2) + .dataExtractor(function(nodeData){ + return {"name": nodeData} + }) + lassoWidget() + } + + } + + + + + return plot +} diff --git a/demos/upset/index.html b/demos/upset/index.html new file mode 100644 index 0000000..71f5e2e --- /dev/null +++ b/demos/upset/index.html @@ -0,0 +1,133 @@ + + + + + UpSet Plot + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + +
+
+
+ + +
+
Rendered with v0.0.3
+ +
+
+ +
+
+
+
+
+ + + + + + + + + + diff --git a/demos/upset/js/v003.js b/demos/upset/js/v003.js new file mode 100644 index 0000000..f08937c --- /dev/null +++ b/demos/upset/js/v003.js @@ -0,0 +1,300 @@ +function upset ( selection ) { + var + data, + namespace='upset' + + + var + upset, + upsetAxis, + setTotalsBarChart, + setTotalsAxis, + intersectionTotalsBarChart, + intersectionTotalsAxis, + intersectionSetsAxis + + plot.data = function(_){return arguments.length ? (data = _, plot) : data; }; + plot.namespace = function(_){return arguments.length ? (namespace = _, plot) : namespace; }; + plot.upset = function(_){return arguments.length ? (upset = _, plot) : upset; }; + plot.upsetAxis = function(_){return arguments.length ? (upsetAxis = _, plot) : upsetAxis; }; + plot.setTotalsBarChart = function(_){return arguments.length ? (setTotalsBarChart = _, plot) : setTotalsBarChart; }; + plot.setTotalsAxis = function(_){return arguments.length ? (setTotalsAxis = _, plot) : setTotalsAxis; }; + plot.intersectionTotalsBarChart = function(_){return arguments.length ? (intersectionTotalsBarChart = _, plot) : intersectionTotalsBarChart; }; + plot.intersectionTotalsAxis = function(_){return arguments.length ? (intersectionTotalsAxis = _, plot) : intersectionTotalsAxis; }; + + + + function plot() { + var + svg = d3sm.safeSelect(selection, 'svg'), + + upsetAxisSelection = d3sm.safeSelect(svg, 'g', namespace+'-upset-chart-axis'), + setTotalsAxisSelection = d3sm.safeSelect(svg, 'g', namespace+'-set-bar-chart-axis'), + intersectionTotalsAxisSelection = d3sm.safeSelect(svg, 'g', namespace+'-intersection-bar-chart-axis'), + intersectionSetsAxisSelection = d3sm.safeSelect(svg, 'g', namespace+'-intersection-bar-set-axis'), + + upsetSelection = d3sm.safeSelect(svg, 'g', namespace+'-upset-chart'), + setTotalsBarChartSelection = d3sm.safeSelect(svg, 'g', namespace+'-set-bar-chart'), + intersectionTotalsBarChartSelection = d3sm.safeSelect(svg, 'g', namespace+'-intersection-bar-chart') + + + + if (upset == undefined) { upset = d3sm.upset(upsetSelection) } + if (upsetAxis == undefined) { upsetAxis = d3sm.axis(upsetAxisSelection) } + if (setTotalsBarChart == undefined) { setTotalsBarChart = d3sm.bar(setTotalsBarChartSelection) } + if (setTotalsAxis == undefined) { setTotalsAxis = d3sm.axis(setTotalsAxisSelection) } + if (intersectionTotalsBarChart == undefined) { intersectionTotalsBarChart = d3sm.bar(intersectionTotalsBarChartSelection) } + if (intersectionTotalsAxis == undefined) { intersectionTotalsAxis = d3sm.axis(intersectionTotalsAxisSelection) } + if (intersectionSetsAxis == undefined) { intersectionSetsAxis = d3sm.axis(intersectionSetsAxisSelection) } + + + + var + setChartSpacer = 10, + intersectionChartSpacer = 10, + + percentages = { + inChart: {w: 0.75, h: 0.5 }, + usChart: {w: 0.75, h: 0.2 }, + stChart: {w: 0.10, h: 0.2 }, + + isAxis : {w: 0.75, h: 0.2 }, + inAxis : {w: 0.15, h: 0.5 }, + usAxis : {w: 0.15, h: 0.2 }, + stAxis : {w: 0.10, h: 0.1 }, + + svg: {w: 1, h: 0.7} + } + + var + page = {w: selection.node().getBoundingClientRect().width, h: window.innerHeight}, + margins = {left: page.w *0.01, right: page.w *0.01, top: page.h*0.01, bottom: page.h*0.01}, + svgRect = { + _w: page.w * percentages.svg.w, + _h: page.h * percentages.svg.h, + w: page.w * percentages.svg.w - margins.left - margins.right, + h: page.h * percentages.svg.h - margins.top - margins.bottom + }, + chartRect = { + w: svgRect.w - setChartSpacer*2, + h: svgRect.h - intersectionChartSpacer*2 + }, + upsetRect = { + x: margins.left + (chartRect.w * percentages.stChart.w) + (chartRect.w * percentages.usAxis.w) + setChartSpacer*2, + y: margins.top + chartRect.h * percentages.inChart.h + (chartRect.h * percentages.isAxis.h) + intersectionChartSpacer, + w: chartRect.w * percentages.usChart.w, + h: chartRect.h * percentages.usChart.h + }, + upsetAxisRect = { + x: margins.left + (chartRect.w * percentages.stChart.w) + (chartRect.w * percentages.usAxis.w) + setChartSpacer, + y: margins.top + chartRect.h * percentages.inChart.h + (chartRect.h * percentages.isAxis.h) + intersectionChartSpacer, + w: chartRect.w * percentages.usAxis.w, + h: chartRect.h * percentages.usAxis.h + }, + setTotalsRect = { + x: margins.left, + y: margins.top + chartRect.h * percentages.inChart.h + (chartRect.h * percentages.isAxis.h) + intersectionChartSpacer, + w: chartRect.w * percentages.stChart.w, + h: chartRect.h * percentages.stChart.h + }, + setTotalsAxisRect = { + x: margins.left, + y: margins.top + (chartRect.h * percentages.inChart.h) + (chartRect.h * percentages.stChart.h) + (chartRect.h * percentages.isAxis.h) + intersectionChartSpacer*2, + w: chartRect.w * percentages.stAxis.w, + h: chartRect.h * percentages.stAxis.h + }, + intersectionTotalsRect = { + x: margins.left + (chartRect.w * percentages.stChart.w) + (chartRect.w * percentages.usAxis.w) + setChartSpacer*2, + y: margins.top + (chartRect.h * percentages.isAxis.h), + w: chartRect.w * percentages.inChart.w, + h: chartRect.h * percentages.inChart.h + }, + intersectionTotalsAxisRect = { + x: margins.left + (chartRect.w * percentages.stChart.w) + (chartRect.w * percentages.usAxis.w) + setChartSpacer, + y: margins.top + (chartRect.h * percentages.isAxis.h), + w: chartRect.w * percentages.inAxis.w, + h: chartRect.h * percentages.inAxis.h + }, + intersectionSetsAxisRect = { + x: margins.left + (chartRect.w * percentages.stChart.w) + (chartRect.w * percentages.usAxis.w) + setChartSpacer + intersectionChartSpacer, + y: margins.top + (chartRect.h * percentages.isAxis.h), + w: chartRect.w * percentages.isAxis.w, + h: chartRect.h * percentages.isAxis.h + } + + + svg.attr('width', svgRect._w).attr('height', svgRect._h) + upsetAxisSelection.attr('transform', 'translate('+upsetAxisRect.x+','+upsetAxisRect.y+')') + setTotalsAxisSelection.attr('transform', 'translate('+setTotalsAxisRect.x+','+setTotalsAxisRect.y+')') + intersectionTotalsAxisSelection.attr('transform', 'translate('+intersectionTotalsAxisRect.x+','+intersectionTotalsAxisRect.y+')') + intersectionSetsAxisSelection.attr('transform', 'translate('+intersectionSetsAxisRect.x+','+intersectionSetsAxisRect.y+')') + + upsetSelection.attr('transform', 'translate('+upsetRect.x+','+upsetRect.y+')') + setTotalsBarChartSelection.attr('transform', 'translate('+setTotalsRect.x+','+setTotalsRect.y+')') + intersectionTotalsBarChartSelection.attr('transform', 'translate('+intersectionTotalsRect.x+','+intersectionTotalsRect.y+')') + + var overflowQ = true + + upset + .data(data) + .spaceX(upsetRect.w) + .spaceY(upsetRect.h) + .xObjectSpacer(0.01) + .yObjectSpacer(0.01) + .maxObjectSize(5) + .minObjectSize(20) + .overflowQ(overflowQ) + .intersectionExtractor(function(k, i){ return data[k].intersection.join(';') }) + .intersectionKeySortingFunction(function(a, b){ + return ( + data[a].intersection.length - data[b].intersection.length + || + upset.intersectionValues().indexOf(data[a].intersection.join(';')) + -upset.intersectionValues().indexOf(data[b].intersection.join(';')) + ) + }) + // .radius(1) + upset() + + upsetAxis + .spaceX(upsetAxisRect.w) + .spaceY(upsetAxisRect.h) + .orient('left') + .categoricalQ(true) + .tickLabels(upset.setValues()) + .overflowQ(overflowQ) + .minObjectSize(upset.minObjectSize()) + .maxObjectSize(upset.maxObjectSize()) + .objectSpacer(upset.yObjectSpacer()) + .tickLabelMargin(30) + + upsetAxis() + + var intersectionTotals = upset.intersectionTotals() + var setTotals = upset.setTotals() + + + intersectionSetsAxis + .tickLabels(upset.intersectionValues()) + .categoricalQ(true) + .spaceX(intersectionSetsAxisRect.w) + .spaceY(intersectionSetsAxisRect.h) + .orient('top') + .namespace('intersection-sets-name') + .overflowQ(overflowQ) + .minObjectSize(upset.minObjectSize()) + .maxObjectSize(upset.maxObjectSize()) + .objectSpacer(upset.xObjectSpacer()) + .objectSize(upset.xObjectSize()) + .spacerSize(upset.xSpacerSize()) + // .tickLabelRotation(-45) + .tickTickLabelSpacer(0) + + intersectionSetsAxis() + + intersectionTotalsBarChart.data(intersectionTotals) + .grouping(upset.intersectionValues()) + .spaceX(intersectionTotalsRect.w) + .spaceY(intersectionTotalsRect.h) + .namespace('intersections') + .orient('horizontal') + .valueExtractor(function(k, i){return intersectionTotals[k]['total']}) + .overflowQ(overflowQ) + .minObjectSize(upset.minObjectSize()) + .maxObjectSize(upset.maxObjectSize()) + .objectSpacer(upset.xObjectSpacer()) + .objectSize(upset.xObjectSize()) + .spacerSize(upset.xSpacerSize()) + + + intersectionTotalsBarChart.tooltip().keys(['Amount','Values']) + .values([ + function(currentData, d){ + return currentData.total + }, + function(currentData, d){ + var l = currentData.values.length + return currentData.values.join(', ').slice(0,100)+ ((l > 100) ? '...': '') + } + ]) + + // intersectionTotalsBarChart.colorFunction().colors(['cyan', 'blue']).colorBy('value') + + // intersectionSetsAxis() + intersectionTotalsBarChart() + + intersectionTotalsAxis + .spaceX(intersectionTotalsAxisRect.w) + .spaceY(intersectionTotalsAxisRect.h) + .orient('left') + .namespace('intersections-axis-left') + .tickValues(intersectionTotalsBarChart.barValues()) + .guideLinesQ(true) + .guidelineSpace(intersectionTotalsRect.w + intersectionChartSpacer) + .overflowQ(overflowQ) + + intersectionTotalsAxis() + + + + setTotalsBarChart.data(setTotals) + .grouping(upset.setValues()) + .spaceX(setTotalsRect.w) + .spaceY(setTotalsRect.h) + .orient('right') + .valueExtractor(function(k, i){return setTotals[k]['total']}) + .namespace('set') + .overflowQ(overflowQ) + .minObjectSize(upset.minObjectSize()) + .maxObjectSize(upset.maxObjectSize()) + .objectSpacer(upset.yObjectSpacer()) + .objectSize(upset.yObjectSize()) + .spacerSize(upset.ySpacerSize()) + .barPercent(0.75) + + setTotalsBarChart() + + + + setTotalsAxis + .spaceX(setTotalsAxisRect.w) + .spaceY(setTotalsAxisRect.h) + .namespace('set-axis-bottom') + .orient('bottom') + .tickValues(setTotalsBarChart.barValues()) + .reverseScaleQ(true) + // .tickLabelRotation(0) + .overflowQ(overflowQ) + + setTotalsAxis() + + + + + + + var zoom = d3sm.multiPlotZoom(upset).eventType('drag').orient('2D') + .xComponents([ + intersectionTotalsBarChart, + intersectionSetsAxis + ]) + .yComponents([ + upsetAxis, + setTotalsBarChart + ]) + var pan = d3.zoom().scaleExtent([1,1]).on('zoom', function(){ zoom.eventType('drag')() }) + upsetSelection.call(pan).on('wheel', function(){ zoom.eventType('wheel')() }) + intersectionTotalsBarChartSelection.call(pan).on('wheel', function(){ zoom.eventType('wheel')() }) + + + + intersectionTotalsBarChartSelection.selectAll('.bar.intersections > rect.bar-rect').on('click', function(d, i){ + console.log( d, i) + }) + // selection.selectAll('text').attr('pointer-events', 'none') + // .attr('pointer-events', 'none') + + } + + return plot +} diff --git a/demos/upset/js/v004.js b/demos/upset/js/v004.js new file mode 100644 index 0000000..dbaec08 --- /dev/null +++ b/demos/upset/js/v004.js @@ -0,0 +1,301 @@ +function upset ( selection ) { + var + data, + namespace='upset' + + + var + upset, + upsetAxis, + setTotalsBarChart, + setTotalsAxis, + intersectionTotalsBarChart, + intersectionTotalsAxis, + intersectionSetsAxis + + plot.data = function(_){return arguments.length ? (data = _, plot) : data; }; + plot.namespace = function(_){return arguments.length ? (namespace = _, plot) : namespace; }; + plot.upset = function(_){return arguments.length ? (upset = _, plot) : upset; }; + plot.upsetAxis = function(_){return arguments.length ? (upsetAxis = _, plot) : upsetAxis; }; + plot.setTotalsBarChart = function(_){return arguments.length ? (setTotalsBarChart = _, plot) : setTotalsBarChart; }; + plot.setTotalsAxis = function(_){return arguments.length ? (setTotalsAxis = _, plot) : setTotalsAxis; }; + plot.intersectionTotalsBarChart = function(_){return arguments.length ? (intersectionTotalsBarChart = _, plot) : intersectionTotalsBarChart; }; + plot.intersectionTotalsAxis = function(_){return arguments.length ? (intersectionTotalsAxis = _, plot) : intersectionTotalsAxis; }; + + + + function plot() { + var + svg = d3sm.utils.sel.safeSelect(selection, 'svg'), + + upsetAxisSelection = d3sm.utils.sel.safeSelect(svg, 'g', namespace+'-upset-chart-axis'), + setTotalsAxisSelection = d3sm.utils.sel.safeSelect(svg, 'g', namespace+'-set-bar-chart-axis'), + intersectionTotalsAxisSelection = d3sm.utils.sel.safeSelect(svg, 'g', namespace+'-intersection-bar-chart-axis'), + intersectionSetsAxisSelection = d3sm.utils.sel.safeSelect(svg, 'g', namespace+'-intersection-bar-set-axis'), + + upsetSelection = d3sm.utils.sel.safeSelect(svg, 'g', namespace+'-upset-chart'), + setTotalsBarChartSelection = d3sm.utils.sel.safeSelect(svg, 'g', namespace+'-set-bar-chart'), + intersectionTotalsBarChartSelection = d3sm.utils.sel.safeSelect(svg, 'g', namespace+'-intersection-bar-chart') + + + + if (upset == undefined) { upset = d3sm.charts.upset(upsetSelection) } + if (upsetAxis == undefined) { upsetAxis = d3sm.axis(upsetAxisSelection) } + if (setTotalsBarChart == undefined) { setTotalsBarChart = d3sm.charts.bar(setTotalsBarChartSelection) } + if (setTotalsAxis == undefined) { setTotalsAxis = d3sm.axis(setTotalsAxisSelection) } + if (intersectionTotalsBarChart == undefined) { intersectionTotalsBarChart = d3sm.charts.bar(intersectionTotalsBarChartSelection) } + if (intersectionTotalsAxis == undefined) { intersectionTotalsAxis = d3sm.axis(intersectionTotalsAxisSelection) } + if (intersectionSetsAxis == undefined) { intersectionSetsAxis = d3sm.axis(intersectionSetsAxisSelection) } + + + + var + setChartSpacer = 10, + intersectionChartSpacer = 10, + + percentages = { + inChart: {w: 0.75, h: 0.5 }, + usChart: {w: 0.75, h: 0.2 }, + stChart: {w: 0.10, h: 0.2 }, + + isAxis : {w: 0.75, h: 0.2 }, + inAxis : {w: 0.15, h: 0.5 }, + usAxis : {w: 0.15, h: 0.2 }, + stAxis : {w: 0.10, h: 0.1 }, + + svg: {w: 1, h: 0.7} + } + + var + page = {w: selection.node().getBoundingClientRect().width, h: window.innerHeight}, + margins = {left: page.w *0.01, right: page.w *0.01, top: page.h*0.01, bottom: page.h*0.01}, + svgRect = { + _w: page.w * percentages.svg.w, + _h: page.h * percentages.svg.h, + w: page.w * percentages.svg.w - margins.left - margins.right, + h: page.h * percentages.svg.h - margins.top - margins.bottom + }, + chartRect = { + w: svgRect.w - setChartSpacer*2, + h: svgRect.h - intersectionChartSpacer*2 + }, + upsetRect = { + x: margins.left + (chartRect.w * percentages.stChart.w) + (chartRect.w * percentages.usAxis.w) + setChartSpacer*2, + y: margins.top + chartRect.h * percentages.inChart.h + (chartRect.h * percentages.isAxis.h) + intersectionChartSpacer, + w: chartRect.w * percentages.usChart.w, + h: chartRect.h * percentages.usChart.h + }, + upsetAxisRect = { + x: margins.left + (chartRect.w * percentages.stChart.w) + (chartRect.w * percentages.usAxis.w) + setChartSpacer, + y: margins.top + chartRect.h * percentages.inChart.h + (chartRect.h * percentages.isAxis.h) + intersectionChartSpacer, + w: chartRect.w * percentages.usAxis.w, + h: chartRect.h * percentages.usAxis.h + }, + setTotalsRect = { + x: margins.left, + y: margins.top + chartRect.h * percentages.inChart.h + (chartRect.h * percentages.isAxis.h) + intersectionChartSpacer, + w: chartRect.w * percentages.stChart.w, + h: chartRect.h * percentages.stChart.h + }, + setTotalsAxisRect = { + x: margins.left, + y: margins.top + (chartRect.h * percentages.inChart.h) + (chartRect.h * percentages.stChart.h) + (chartRect.h * percentages.isAxis.h) + intersectionChartSpacer*2, + w: chartRect.w * percentages.stAxis.w, + h: chartRect.h * percentages.stAxis.h + }, + intersectionTotalsRect = { + x: margins.left + (chartRect.w * percentages.stChart.w) + (chartRect.w * percentages.usAxis.w) + setChartSpacer*2, + y: margins.top + (chartRect.h * percentages.isAxis.h), + w: chartRect.w * percentages.inChart.w, + h: chartRect.h * percentages.inChart.h + }, + intersectionTotalsAxisRect = { + x: margins.left + (chartRect.w * percentages.stChart.w) + (chartRect.w * percentages.usAxis.w) + setChartSpacer, + y: margins.top + (chartRect.h * percentages.isAxis.h), + w: chartRect.w * percentages.inAxis.w, + h: chartRect.h * percentages.inAxis.h + }, + intersectionSetsAxisRect = { + x: margins.left + (chartRect.w * percentages.stChart.w) + (chartRect.w * percentages.usAxis.w) + setChartSpacer + intersectionChartSpacer, + y: margins.top + (chartRect.h * percentages.isAxis.h), + w: chartRect.w * percentages.isAxis.w, + h: chartRect.h * percentages.isAxis.h + } + + + svg.attr('width', svgRect._w).attr('height', svgRect._h) + upsetAxisSelection.attr('transform', 'translate('+upsetAxisRect.x+','+upsetAxisRect.y+')') + setTotalsAxisSelection.attr('transform', 'translate('+setTotalsAxisRect.x+','+setTotalsAxisRect.y+')') + intersectionTotalsAxisSelection.attr('transform', 'translate('+intersectionTotalsAxisRect.x+','+intersectionTotalsAxisRect.y+')') + intersectionSetsAxisSelection.attr('transform', 'translate('+intersectionSetsAxisRect.x+','+intersectionSetsAxisRect.y+')') + + upsetSelection.attr('transform', 'translate('+upsetRect.x+','+upsetRect.y+')') + setTotalsBarChartSelection.attr('transform', 'translate('+setTotalsRect.x+','+setTotalsRect.y+')') + intersectionTotalsBarChartSelection.attr('transform', 'translate('+intersectionTotalsRect.x+','+intersectionTotalsRect.y+')') + + var overflowQ = true + + upset + .data(data) + .spaceX(upsetRect.w) + .spaceY(upsetRect.h) + .xObjectSpacer(0.01) + .yObjectSpacer(0.01) + .maxObjectSize(5) + .minObjectSize(20) + .overflowQ(overflowQ) + .intersectionExtractor(function(k, i){ return data[k].intersection.join(';') }) + .intersectionKeySortingFunction(function(a, b){ + return ( + data[a].intersection.length - data[b].intersection.length + || + upset.intersectionValues().indexOf(data[a].intersection.join(';')) + -upset.intersectionValues().indexOf(data[b].intersection.join(';')) + ) + }) + // .radius(1) + upset() + + upsetAxis + .spaceX(upsetAxisRect.w) + .spaceY(upsetAxisRect.h) + .orient('left') + .categoricalQ(true) + .tickLabels(upset.setValues()) + .overflowQ(overflowQ) + .minObjectSize(upset.minObjectSize()) + .maxObjectSize(upset.maxObjectSize()) + .objectSpacer(upset.yObjectSpacer()) + .tickLabelMargin(30) + + upsetAxis() + + var intersectionTotals = upset.intersectionTotals() + var setTotals = upset.setTotals() + + + intersectionSetsAxis + .tickLabels(upset.intersectionValues()) + .categoricalQ(true) + .spaceX(intersectionSetsAxisRect.w) + .spaceY(intersectionSetsAxisRect.h) + .orient('top') + .namespace('intersection-sets-name') + .overflowQ(overflowQ) + .minObjectSize(upset.minObjectSize()) + .maxObjectSize(upset.maxObjectSize()) + .objectSpacer(upset.xObjectSpacer()) + .objectSize(upset.xObjectSize()) + .spacerSize(upset.xSpacerSize()) + // .tickLabelRotation(-45) + .tickTickLabelSpacer(0) + + intersectionSetsAxis() + + intersectionTotalsBarChart.data(intersectionTotals) + .grouping(upset.intersectionValues()) + .spaceX(intersectionTotalsRect.w) + .spaceY(intersectionTotalsRect.h) + .namespace('intersections') + .orient('horizontal') + .valueExtractor(function(k, i){return intersectionTotals[k]['total']}) + .overflowQ(overflowQ) + .minObjectSize(upset.minObjectSize()) + .maxObjectSize(upset.maxObjectSize()) + .objectSpacer(upset.xObjectSpacer()) + .objectSize(upset.xObjectSize()) + .spacerSize(upset.xSpacerSize()) + + + intersectionTotalsBarChart.tooltip().keys(['Amount','Values']) + .values([ + function(currentData, d){ + return currentData.total + }, + function(currentData, d){ + var l = currentData.values.length + // return currentData + return currentData.values.join(', ').slice(0,100)+ ((l > 100) ? '...': '') + } + ]) + + // intersectionTotalsBarChart.colorFunction().colors(['cyan', 'blue']).colorBy('value') + + // intersectionSetsAxis() + intersectionTotalsBarChart() + + intersectionTotalsAxis + .spaceX(intersectionTotalsAxisRect.w) + .spaceY(intersectionTotalsAxisRect.h) + .orient('left') + .namespace('intersections-axis-left') + .tickValues(intersectionTotalsBarChart.barValues()) + .guideLinesQ(true) + .guidelineSpace(intersectionTotalsRect.w + intersectionChartSpacer) + .overflowQ(overflowQ) + + intersectionTotalsAxis() + + + + setTotalsBarChart.data(setTotals) + .grouping(upset.setValues()) + .spaceX(setTotalsRect.w) + .spaceY(setTotalsRect.h) + .orient('right') + .valueExtractor(function(k, i){return setTotals[k]['total']}) + .namespace('set') + .overflowQ(overflowQ) + .minObjectSize(upset.minObjectSize()) + .maxObjectSize(upset.maxObjectSize()) + .objectSpacer(upset.yObjectSpacer()) + .objectSize(upset.yObjectSize()) + .spacerSize(upset.ySpacerSize()) + .barPercent(0.75) + + setTotalsBarChart() + + + + setTotalsAxis + .spaceX(setTotalsAxisRect.w) + .spaceY(setTotalsAxisRect.h) + .namespace('set-axis-bottom') + .orient('bottom') + .tickValues(setTotalsBarChart.barValues()) + .reverseScaleQ(true) + // .tickLabelRotation(0) + .overflowQ(overflowQ) + + setTotalsAxis() + + + + + + + var zoom = d3sm.aux.multiPlotZoom(upset).eventType('drag').orient('2D') + .xComponents([ + intersectionTotalsBarChart, + intersectionSetsAxis + ]) + .yComponents([ + upsetAxis, + setTotalsBarChart + ]) + var pan = d3.zoom().scaleExtent([1,1]).on('zoom', function(){ zoom.eventType('drag')() }) + upsetSelection.call(pan).on('wheel', function(){ zoom.eventType('wheel')() }) + intersectionTotalsBarChartSelection.call(pan).on('wheel', function(){ zoom.eventType('wheel')() }) + + + + intersectionTotalsBarChartSelection.selectAll('.bar.intersections > rect.bar-rect').on('click', function(d, i){ + console.log( d, i) + }) + // selection.selectAll('text').attr('pointer-events', 'none') + // .attr('pointer-events', 'none') + + } + + return plot +} diff --git a/demos/vertical-violins/index.html b/demos/vertical-violins/index.html new file mode 100644 index 0000000..6f33aef --- /dev/null +++ b/demos/vertical-violins/index.html @@ -0,0 +1,158 @@ + + + + + Basic Violins + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/dist/d3sm.esm.js b/dist/d3sm.esm.js index b9ee62a..76be704 100644 --- a/dist/d3sm.esm.js +++ b/dist/d3sm.esm.js @@ -58,7 +58,7 @@ function last( array ) { return array[array.length-1]; } * @param {number[]} array of numerical values * @returns {number} sum over elements in array */ -function total$1( array ) { return array.reduce((a, b) => a + b, 0) } +function total( array ) { return array.reduce((a, b) => a + b, 0) } /** * Removes duplicates in array * @param {Array} array of items @@ -212,7 +212,7 @@ var arr = /*#__PURE__*/Object.freeze({ hasQ: hasQ, first: first, last: last, - total: total$1, + total: total, unique: unique, get: get, listOfListsQ: listOfListsQ, @@ -435,150 +435,6 @@ var math = /*#__PURE__*/Object.freeze({ spacersNeededAtEachLevel: spacersNeededAtEachLevel }); -function resizeDebounce(f, wait) { - var resize = debounce(function(){f();},wait); - window.addEventListener('resize', resize); -} - -function debounce(func, wait, immediate) { - var timeout; - return function() { - var context = this, args = arguments; - var later = function() { - timeout = null; - if (!immediate) func.apply(context, args); - }; - var callNow = immediate && !timeout; - clearTimeout(timeout); - timeout = setTimeout(later, wait); - if (callNow) func.apply(context, args); - }; -} - -function setupStandardChartContainers( - selection, - namespace, - container, - margins={top:0.01, bottom:0.01, left:0.01, right:0.01}, - svg={w:0.8, h:0.6}, // percent of container space for svg - axes={y:0.1, x:0.1}, // percent of container space for axes, - leg={x:0, margin:0, pos:'left'} // absolute width of legend and space on either size -) -{ - if (container == undefined) {container = {w:window.innerWidth, h:window.Height};} - // SVG width and height - - var svgSpace = { - w: container.w * svg.w, - h: container.h * svg.h - }; - - var margPx = { - top: margins.top * svgSpace.h, - bottom: margins.bottom * svgSpace.h, - left: margins.left * svgSpace.w, - right: margins.right * svgSpace.w - }, - - - - // Space after removing margins - chartSpace = { - w: svgSpace.w - margPx.left - margPx.right, - h: svgSpace.h - margPx.top - margPx.bottom - }, - - // main dimension of x and y axies - // e.g. defines how tall x axis is as length is determined by plotRect.w - axesSpace = { - x: chartSpace.h * axes.x, - y: chartSpace.w * axes.y - }, - - // space left for drawing the chart properly (e.g. bars, violins, etc) - drawingSpace = { - x: chartSpace.w - axesSpace.y - leg.x - 2*leg.margin, - y: chartSpace.h - axesSpace.x - }, - - - legRect = { - x: leg.margin + margPx.left + (leg.pos == 'left' ? 0 : drawingSpace.x + axesSpace.y), - y: margPx.top, // this is soomehow getting calculated incorectly - w: leg.x, - h: drawingSpace.y - }, - - yAxisRect = { - x: axesSpace.y + margPx.left + (leg.pos == 'left' ? leg.x + 2*leg.margin : 0), - y: margPx.top, - w: axesSpace.y, - h: drawingSpace.y - }, - - plotRect = { - x: axesSpace.y + margPx.left + (leg.pos == 'left' ? leg.x + 2*leg.margin : 0), - y: margPx.top, - w: drawingSpace.x, - h: drawingSpace.y - }, - - xAxisRect = { - x: axesSpace.y + margPx.left + (leg.pos == 'left' ? leg.x + 2*leg.margin : 0), - y: margPx.top + drawingSpace.y, - w: drawingSpace.x, - h: axesSpace.x - }; - - - - container = d3sm.safeSelect(selection, 'svg', namespace) - .style('width', svgSpace.w+'px') - .style('height', svgSpace.h+'px'); - - var axes = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'axes')); - - var leg = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'legend')) - .attr('transform', "translate("+legRect.x+","+legRect.y+")"); - - var plot = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'plot')) - .attr('transform', "translate("+plotRect.x+","+plotRect.y+")"); - - var xAxis = d3sm.safeSelect(axes, 'g', d3sm.hypenate(namespace, 'x-axis')) - .attr('transform', "translate("+xAxisRect.x+","+xAxisRect.y+")"); - - var yAxis = d3sm.safeSelect(axes, 'g', d3sm.hypenate(namespace, 'y-axis')) - .attr('transform', "translate("+yAxisRect.x+","+yAxisRect.y+")"); - - return { - svg: { - selection: container, - rect: svgSpace - }, - plot: { - selection: plot, - rect: plotRect - }, - xAxis: { - selection: xAxis, - rect: xAxisRect - }, - yAxis: { - selection: yAxis, - rect: yAxisRect - }, - legend: { - selection: leg, - rect: legRect - } - } -} - -var misc = /*#__PURE__*/Object.freeze({ - resizeDebounce: resizeDebounce, - setupStandardChartContainers: setupStandardChartContainers -}); - /** * Hypenates all strings together * @param {string[]} arguments @@ -652,12 +508,12 @@ function getTranslation(transform) { * @param {Element} element * @returns {Element | undefined} */ -function getContainingSVG$1(element) { +function getContainingSVG(element) { var parent = element.parentElement; var tag = parent.tagName.toLowerCase(); if (tag === 'svg') { return parent; } if (tag === 'html') { return undefined; } - return getContainingSVG$1(parent); + return getContainingSVG(parent); } @@ -748,100 +604,244 @@ function setupContainer(selection, namespace, rect, fill) { var sel = /*#__PURE__*/Object.freeze({ getTranslation: getTranslation, - getContainingSVG: getContainingSVG$1, + getContainingSVG: getContainingSVG, safeSelect: safeSelect, cpRect: cpRect, bgRect: bgRect, setupContainer: setupContainer }); -/** - * calls console.group if d3sm.debugQ == true - * @param {string} name of the group - * @returns {undefined} - */ -function consoleGroup(name) { - if (window.d3sm.debugQ === true){ - console.group(name); - } +function resizeDebounce(f, wait) { + var resize = debounce(function(){f();},wait); + window.addEventListener('resize', resize); } -/** - * calls console.groupEnd if d3sm.debugQ == true - * @returns {undefined} - */ -function consoleGroupEnd() { - if (window.d3sm.debugQ === true){ - console.groupEnd(); - } +function debounce(func, wait, immediate) { + var timeout; + return function() { + var context = this, args = arguments; + var later = function() { + timeout = null; + if (!immediate) func.apply(context, args); + }; + var callNow = immediate && !timeout; + clearTimeout(timeout); + timeout = setTimeout(later, wait); + if (callNow) func.apply(context, args); + }; } -/** - * Calls console.log if d3sm.debugQ == true - * @param {string} func name of the function logging - * @param {string} msg to log - * @param {Object} data to be logged along side the message - * @returns {undefined} - */ -function log(func, msg, data) { - if (window.d3sm.debugQ === true){ - console.log( - `%c[d3sm::${func}]:\t${msg}`, - [ - 'background: #6cd1ef', - 'border-radius: 5000px', - 'padding: 0px 2px', - 'font-size: 14px' - ].join(';') - ); - console.table(data); - // console.trace() - } -} +function setupStandardChartContainers( + selection, + namespace, + container, + margins={top:0.01, bottom:0.01, left:0.01, right:0.01}, + svg={w:0.8, h:0.6}, // percent of container space for svg + axes={y:0.1, x:0.1}, // percent of container space for axes, + leg={x:0, margin:0, pos:'left'} // absolute width of legend and space on either size +) +{ + if (container == undefined) {container = {w:window.innerWidth, h:window.Height};} + // SVG width and height -/** - * Calls console.warn if d3sm.debugQ == true - * @param {string} func name of the function warning - * @param {string} msg to display - * @param {Object} data to be displayed along side the message - * @returns {undefined} - */ -function warn(func, msg, data) { - if (window.d3sm.debugQ === true) - console.warn( - `%c[d3sm::${func}]:\t${msg}`, - [ - 'background: #ffd53e', - 'border-radius: 5000px', - 'padding: 0px 2px', - 'font-size: 14px' - ].join(';') - ); - console.table(data); -} -/** - * Calls the console.info if d3sm.debugQ == true - * @param {string} func name of the function providing info - * @param {string} msg to display - * @param {Object} data to be displayed along side the message - * @returns {undefined} - */ -function info(func, msg, data) { - if (window.d3sm.debugQ) - console.info( - `%c[d3sm::${func}]:\t${msg}`, - [ - 'background: #009ccd', - 'border-radius: 5000px', - 'padding: 0px 2px', - 'font-size: 14px' - ].join(';') - ); - console.table(data); -} + var svgSpace = { + w: container.w * svg.w, + h: container.h * svg.h + }; + var margPx = { + top: margins.top * svgSpace.h, + bottom: margins.bottom * svgSpace.h, + left: margins.left * svgSpace.w, + right: margins.right * svgSpace.w + }, -/** + + + // Space after removing margins + chartSpace = { + w: svgSpace.w - margPx.left - margPx.right, + h: svgSpace.h - margPx.top - margPx.bottom + }, + + // main dimension of x and y axies + // e.g. defines how tall x axis is as length is determined by plotRect.w + axesSpace = { + x: chartSpace.h * axes.x, + y: chartSpace.w * axes.y + }, + + // space left for drawing the chart properly (e.g. bars, violins, etc) + drawingSpace = { + x: chartSpace.w - axesSpace.y - leg.x - 2*leg.margin, + y: chartSpace.h - axesSpace.x + }, + + + legRect = { + x: leg.margin + margPx.left + (leg.pos == 'left' ? 0 : drawingSpace.x + axesSpace.y), + y: margPx.top, // this is soomehow getting calculated incorectly + w: leg.x, + h: drawingSpace.y + }, + + yAxisRect = { + x: axesSpace.y + margPx.left + (leg.pos == 'left' ? leg.x + 2*leg.margin : 0), + y: margPx.top, + w: axesSpace.y, + h: drawingSpace.y + }, + + plotRect = { + x: axesSpace.y + margPx.left + (leg.pos == 'left' ? leg.x + 2*leg.margin : 0), + y: margPx.top, + w: drawingSpace.x, + h: drawingSpace.y + }, + + xAxisRect = { + x: axesSpace.y + margPx.left + (leg.pos == 'left' ? leg.x + 2*leg.margin : 0), + y: margPx.top + drawingSpace.y, + w: drawingSpace.x, + h: axesSpace.x + }; + + + + container = safeSelect(selection, 'svg', namespace) + .style('width', svgSpace.w+'px') + .style('height', svgSpace.h+'px'); + + var axes = safeSelect(container, 'g', hypenate(namespace, 'axes')); + + var leg = safeSelect(container, 'g', hypenate(namespace, 'legend')) + .attr('transform', "translate("+legRect.x+","+legRect.y+")"); + + var plot = safeSelect(container, 'g', hypenate(namespace, 'plot')) + .attr('transform', "translate("+plotRect.x+","+plotRect.y+")"); + + var xAxis = safeSelect(axes, 'g', hypenate(namespace, 'x-axis')) + .attr('transform', "translate("+xAxisRect.x+","+xAxisRect.y+")"); + + var yAxis = safeSelect(axes, 'g', hypenate(namespace, 'y-axis')) + .attr('transform', "translate("+yAxisRect.x+","+yAxisRect.y+")"); + + return { + svg: { + selection: container, + rect: svgSpace + }, + plot: { + selection: plot, + rect: plotRect + }, + xAxis: { + selection: xAxis, + rect: xAxisRect + }, + yAxis: { + selection: yAxis, + rect: yAxisRect + }, + legend: { + selection: leg, + rect: legRect + } + } +} + +var misc = /*#__PURE__*/Object.freeze({ + resizeDebounce: resizeDebounce, + setupStandardChartContainers: setupStandardChartContainers +}); + +/** + * calls console.group if d3sm.debugQ == true + * @param {string} name of the group + * @returns {undefined} + */ +function consoleGroup(name) { + if (window.d3sm.debugQ === true){ + console.group(name); + } +} + +/** + * calls console.groupEnd if d3sm.debugQ == true + * @returns {undefined} + */ +function consoleGroupEnd() { + if (window.d3sm.debugQ === true){ + console.groupEnd(); + } +} + +/** + * Calls console.log if d3sm.debugQ == true + * @param {string} func name of the function logging + * @param {string} msg to log + * @param {Object} data to be logged along side the message + * @returns {undefined} + */ +function log(func, msg, data) { + if (window.d3sm.debugQ === true){ + console.log( + `%c[d3sm::${func}]:\t${msg}`, + [ + 'background: #6cd1ef', + 'border-radius: 5000px', + 'padding: 0px 2px', + 'font-size: 14px' + ].join(';') + ); + console.table(data); + // console.trace() + } +} + +/** + * Calls console.warn if d3sm.debugQ == true + * @param {string} func name of the function warning + * @param {string} msg to display + * @param {Object} data to be displayed along side the message + * @returns {undefined} + */ +function warn(func, msg, data) { + if (window.d3sm.debugQ === true) + console.warn( + `%c[d3sm::${func}]:\t${msg}`, + [ + 'background: #ffd53e', + 'border-radius: 5000px', + 'padding: 0px 2px', + 'font-size: 14px' + ].join(';') + ); + console.table(data); +} +/** + * Calls the console.info if d3sm.debugQ == true + * @param {string} func name of the function providing info + * @param {string} msg to display + * @param {Object} data to be displayed along side the message + * @returns {undefined} + */ +function info(func, msg, data) { + if (window.d3sm.debugQ) + console.info( + `%c[d3sm::${func}]:\t${msg}`, + [ + 'background: #009ccd', + 'border-radius: 5000px', + 'padding: 0px 2px', + 'font-size: 14px' + ].join(';') + ); + console.table(data); +} + + +/** * Calls console.error if d3sm.debugQ == true * @param {string} func name of the function which sends the error * @param {string} msg to display @@ -906,7 +906,7 @@ var paths = /*#__PURE__*/Object.freeze({ whiskerPath: whiskerPath }); -let utils$1 = { +let utils = { arr, color, math, misc, sel, str, con, paths }; @@ -2016,7 +2016,7 @@ function axis ( selection ) { function axis () { // for convenience in handling orientation specific values - var horizontalQ = utils$1.arr.hasQ(['top', 'bottom', 'horizontal'], orient) ? true : false; + var horizontalQ = utils.arr.hasQ(['top', 'bottom', 'horizontal'], orient) ? true : false; var verticalQ = !horizontalQ; // background cliping rectangle @@ -2047,7 +2047,7 @@ function axis ( selection ) { } - var container = utils$1.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); + var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); // defaults for text-anchor and text rotation if (orient == 'top') { @@ -2085,12 +2085,12 @@ function axis ( selection ) { : (grouping == undefined) ? (numberOfTicks != undefined) // ? (tickValues.length < numberOfTicks) - ? (utils$1.math.tickRange(...d3.extent(tickValues), numberOfTicks)) + ? (utils.math.tickRange(...d3.extent(tickValues), numberOfTicks)) : tickValues : grouping; - var flatTickData = utils$1.arr.flatten(tickData); + var flatTickData = utils.arr.flatten(tickData); var numberOfObjects = flatTickData.length; var space = horizontalQ ? spaceX : spaceY; var extent = d3.extent(flatTickData); @@ -2114,16 +2114,16 @@ function axis ( selection ) { // calculate object size if needed objectSize = (objectSize == undefined) - ? utils$1.math.calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ) + ? utils.math.calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ) : objectSize; // calculate spacer size if needed spacerSize = (spacerSize == undefined) - ? utils$1.math.calculateWidthOfSpacer(flatTickData, space, objectSize, numberOfObjects, objectSpacer, overflowQ) + ? utils.math.calculateWidthOfSpacer(flatTickData, space, objectSize, numberOfObjects, objectSpacer, overflowQ) : spacerSize; - var objClass = utils$1.str.hypenate(namespace, categoricalQ ? objectClass+'-categorical' : objectClass); + var objClass = utils.str.hypenate(namespace, categoricalQ ? objectClass+'-categorical' : objectClass); var spacerFunction = groupingSpacer() .horizontalQ(horizontalQ).scale(scale).moveby((categoricalQ?'category':'scale')).numberOfObjects(numberOfObjects) @@ -2188,11 +2188,11 @@ function axis ( selection ) { - var labelNameGroup = utils$1.sel.safeSelect(selection, 'g', utils$1.str.hypenate(namespace,'axis-name')); + var labelNameGroup = utils.sel.safeSelect(selection, 'g', utils.str.hypenate(namespace,'axis-name')); - var labelElement = utils$1.sel.safeSelect(labelNameGroup, 'text', utils$1.str.hypenate(namespace, 'name')); + var labelElement = utils.sel.safeSelect(labelNameGroup, 'text', utils.str.hypenate(namespace, 'name')); if (labelElement != undefined) { labelElement.text(label); @@ -2257,7 +2257,7 @@ function axis ( selection ) { var that = d3.select(this).style('opacity', 1); // make and move tick - var tick = utils$1.sel.safeSelect(that, 'line', utils$1.str.hypenate(namespace,'tick')) + var tick = utils.sel.safeSelect(that, 'line', utils.str.hypenate(namespace,'tick')) .attr("x1", 0) .attr("x2", horizontalQ ? 0 : orient == "left" ? -tickLength : tickLength) .attr("y1", 0) @@ -2273,10 +2273,10 @@ function axis ( selection ) { }); // make and move label - var label = utils$1.sel.safeSelect(that, 'text', utils$1.str.hypenate(namespace,'label')) + var label = utils.sel.safeSelect(that, 'text', utils.str.hypenate(namespace,'label')) .text(function(d, i){ - var s = typeof d == 'number' ? utils$1.math.round(d, roundTo) : d; - s = utils$1.str.truncateString(String(s), (horizontalQ ? spaceY : spaceX) - tickLength-tickLabelMargin-tickTickLabelSpacer, tickLabelFontSize * 0.45); + var s = typeof d == 'number' ? utils.math.round(d, roundTo) : d; + s = utils.str.truncateString(String(s), (horizontalQ ? spaceY : spaceX) - tickLength-tickLabelMargin-tickTickLabelSpacer, tickLabelFontSize * 0.45); return s }) .attr('font-size', tickLabelFontSize) @@ -2328,7 +2328,7 @@ function axis ( selection ) { // add guidlines as needed if (guideLinesQ) { - var gline = utils$1.sel.safeSelect(that, 'line', utils$1.str.hypenate(namespace, 'guideline')) + var gline = utils.sel.safeSelect(that, 'line', utils.str.hypenate(namespace, 'guideline')) .transition().duration(transitionDuration).ease(easeFunc) .attr("x1", 0) .attr("x2", horizontalQ ? 0 : orient == "left" ? guidelineSpace : -guidelineSpace) @@ -2341,15 +2341,15 @@ function axis ( selection ) { t = 'translate('+x+','+y+')'; return t }); - } else { that.select('line.'+utils$1.str.hypenate(namespace, 'guideline')).remove(); } + } else { that.select('line.'+utils.str.hypenate(namespace, 'guideline')).remove(); } }); // apply alternating guidline thickness if (guideLinesQ) { - container.selectAll('.'+utils$1.str.hypenate(namespace,'guideline')) + container.selectAll('.'+utils.str.hypenate(namespace,'guideline')) .attr('stroke', function(d, i){ - if (i % 2 == 0) { return utils$1.color.modifyHexidecimalColorLuminance(guideLineStroke, 0.8) } + if (i % 2 == 0) { return utils.color.modifyHexidecimalColorLuminance(guideLineStroke, 0.8) } return guideLineStroke }) .attr('stroke-width', function(d, i){ @@ -2363,7 +2363,7 @@ function axis ( selection ) { /*************************************************************************** ** Make the line of the axis ***************************************************************************/ - var line = utils$1.sel.safeSelect(selection, 'path', utils$1.str.hypenate(namespace,'line')) + var line = utils.sel.safeSelect(selection, 'path', utils.str.hypenate(namespace,'line')) // .attr('x1', 0) // .attr('x2', horizontalQ ? spaceX : 0) // .attr('y1', 0) @@ -2383,21 +2383,21 @@ function axis ( selection ) { // hover of label show full text label in case it is truncated function labelHover(d, i){ var t = d3.select(this).style('fill', 'red'); - d3.select(t.node().parentNode).select("line."+utils$1.str.hypenate(namespace,'tick')) + d3.select(t.node().parentNode).select("line."+utils.str.hypenate(namespace,'tick')) .attr("stroke", 'red') .attr("stroke-width", tickStrokeWidth*2); if (guideLinesQ) { - d3.select(t.node().parentNode).select('line.'+utils$1.str.hypenate(namespace, 'guideline')) + d3.select(t.node().parentNode).select('line.'+utils.str.hypenate(namespace, 'guideline')) .attr('stroke', 'red') .attr('stroke-width', guideLineStrokeWidth*2); } - var s = typeof d == 'number' ? utils$1.math.round(d, roundTo) : d; + var s = typeof d == 'number' ? utils.math.round(d, roundTo) : d; var m = d3.mouse(d3.select('html').node()); - var div = utils$1.sel.safeSelect(d3.select('body'), 'div', utils$1.str.hypenate(namespace,'guideline-tooltip')) - .attr('id', utils$1.str.hypenate(namespace,'guideline-tooltip')) + var div = utils.sel.safeSelect(d3.select('body'), 'div', utils.str.hypenate(namespace,'guideline-tooltip')) + .attr('id', utils.str.hypenate(namespace,'guideline-tooltip')) .style('position', 'absolute') .style('left', (d3.event.pageX+15)+'px') .style('top', (d3.event.pageY+15)+'px') @@ -2414,7 +2414,7 @@ function axis ( selection ) { .style('border-style', 'solid') .style('border-width', 2); - var text = utils$1.sel.safeSelect(div, 'div') + var text = utils.sel.safeSelect(div, 'div') .text(tickLabelOnHoverFunc(s, i)) .style('color', 'black') .style('align-self', 'center'); @@ -2427,15 +2427,15 @@ function axis ( selection ) { function labelHoverOff(d, i){ var t = d3.select(this).style('fill', 'black'); - d3.select(t.node().parentNode).select("line."+utils$1.str.hypenate(namespace,'tick')) + d3.select(t.node().parentNode).select("line."+utils.str.hypenate(namespace,'tick')) .attr("stroke", tickStroke) .attr("stroke-width", tickStrokeWidth); if (guideLinesQ) { - var gline = d3.select(t.node().parentNode).select('line.'+utils$1.str.hypenate(namespace, 'guideline')); + var gline = d3.select(t.node().parentNode).select('line.'+utils.str.hypenate(namespace, 'guideline')); var minorQ = gline.attr('minor'); gline.attr('stroke', function(d, ii){ - if (minorQ == 'true') { return utils$1.color.modifyHexidecimalColorLuminance(guideLineStroke, 0.8) } + if (minorQ == 'true') { return utils.color.modifyHexidecimalColorLuminance(guideLineStroke, 0.8) } return guideLineStroke }) .attr('stroke-width', function(d, ii){ @@ -2443,7 +2443,7 @@ function axis ( selection ) { return guideLineStrokeWidth }); } - d3.select("#"+utils$1.str.hypenate(namespace,'guideline-tooltip')).remove(); + d3.select("#"+utils.str.hypenate(namespace,'guideline-tooltip')).remove(); } @@ -3390,7 +3390,7 @@ function scatter ( selection ) { function scatter() { // background cliping rectangle var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; - var container = utils$1.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); + var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); pointKeys = d3.keys(data); @@ -3969,7 +3969,7 @@ function bar ( selection ) { // background cliping rectangle var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; - var container = utils$1.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); + var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); // to prevent re-calculation and getters to be passed to axes barKeys = d3.keys(data); @@ -3978,7 +3978,7 @@ function bar ( selection ) { // if grouping is undefined sort barKeys by sortingFunction var ordered = (grouping == undefined) ? barKeys.sort(sortingFunction) : grouping; // ordered might be nested depending on grouping - barKeys = utils$1.arr.flatten(ordered); + barKeys = utils.arr.flatten(ordered); var numberOfObjects = barKeys.length; var extent = [Math.min(...barValues) - domainPadding,Math.max(...barValues) + domainPadding]; @@ -3996,12 +3996,12 @@ function bar ( selection ) { var space = horizontalQ ? spaceX : spaceY; // calculate object size objectSize = (objectSize == undefined) - ? utils$1.math.calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ) + ? utils.math.calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ) : objectSize; // calculate spacer size if needed spacerSize = (spacerSize == undefined) - ? utils$1.math.calculateWidthOfSpacer(barKeys, space, objectSize, numberOfObjects, objectSpacer, overflowQ) + ? utils.math.calculateWidthOfSpacer(barKeys, space, objectSize, numberOfObjects, objectSpacer, overflowQ) : spacerSize; // make the nested groups var spacerFunction = groupingSpacer() @@ -4068,7 +4068,7 @@ function bar ( selection ) { strokeColor = colorFunction$$1(key, value, i, 'stroke'); - var bar = utils$1.sel.safeSelect(t, 'rect', 'bar-rect'); + var bar = utils.sel.safeSelect(t, 'rect', 'bar-rect'); if (bar.attr('transform') == undefined) { bar.attr('transform', function(d, i) { @@ -4798,19 +4798,19 @@ function bubbleHeatmap( selection ) { function bhm() { var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; - var container = utils$1.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); + var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); cellKeys = d3.keys(data); cellKeys.sort(function(a, b){ return xKeySortingFunction(a, b) || yKeySortingFunction(a, b) }); - utils$1.con.log('bubbleHeatmap', 'cells are sorted by', cellKeys); + utils.con.log('bubbleHeatmap', 'cells are sorted by', cellKeys); - xValues = utils$1.arr.unique(cellKeys.map(xExtractor)); - yValues = utils$1.arr.unique(cellKeys.map(yExtractor)); - rValues = utils$1.arr.unique(cellKeys.map(rExtractor)); - vValues = utils$1.arr.unique(cellKeys.map(vExtractor)); - utils$1.con.log('bubbleHeatmap', 'x and y keys are', {x: xValues, y:yValues}); + xValues = utils.arr.unique(cellKeys.map(xExtractor)); + yValues = utils.arr.unique(cellKeys.map(yExtractor)); + rValues = utils.arr.unique(cellKeys.map(rExtractor)); + vValues = utils.arr.unique(cellKeys.map(vExtractor)); + utils.con.log('bubbleHeatmap', 'x and y keys are', {x: xValues, y:yValues}); var xDim = xValues.length, yDim = yValues.length; @@ -4819,11 +4819,11 @@ function bubbleHeatmap( selection ) { var extent = [Math.min(...rValues) - domainPadding,Math.max(...rValues) + domainPadding]; - ySize = utils$1.math.calculateWidthOfObject(spaceY, yDim, minObjectSize, maxObjectSize, objectSpacer, overflowQ); - xSize = utils$1.math.calculateWidthOfObject(spaceX, xDim, minObjectSize, maxObjectSize, objectSpacer, overflowQ); - ySpacerSize = utils$1.math.calculateWidthOfSpacer(yValues, spaceY, ySize, yDim, objectSpacer, overflowQ); - xSpacerSize = utils$1.math.calculateWidthOfSpacer(xValues, spaceX, xSize, xDim, objectSpacer, overflowQ); - utils$1.con.log('bubbleHeatmap', 'size of', {x: xSize, y: ySize}); + ySize = utils.math.calculateWidthOfObject(spaceY, yDim, minObjectSize, maxObjectSize, objectSpacer, overflowQ); + xSize = utils.math.calculateWidthOfObject(spaceX, xDim, minObjectSize, maxObjectSize, objectSpacer, overflowQ); + ySpacerSize = utils.math.calculateWidthOfSpacer(yValues, spaceY, ySize, yDim, objectSpacer, overflowQ); + xSpacerSize = utils.math.calculateWidthOfSpacer(xValues, spaceX, xSize, xDim, objectSpacer, overflowQ); + utils.con.log('bubbleHeatmap', 'size of', {x: xSize, y: ySize}); scale.domain(extent).range([Math.min(minObjectSize/2, Math.min(ySize, xSize)/2), Math.min(ySize, xSize)/2]); @@ -4831,7 +4831,7 @@ function bubbleHeatmap( selection ) { var ySpacer = groupingSpacer() .horizontalQ(false) .moveby('category').numberOfObjects(yDim) - .objectClass(utils$1.str.hypenate(objectClass, 'row')) + .objectClass(utils.str.hypenate(objectClass, 'row')) .objectSize(ySize).spacerSize(ySpacerSize) .transitionDuration(transitionDuration).easeFunc(easeFunc) .namespace('row'); @@ -4845,7 +4845,7 @@ function bubbleHeatmap( selection ) { ySpacer(container, yValues, 0); - container.selectAll('g.'+utils$1.str.hypenate(objectClass, 'row')) + container.selectAll('g.'+utils.str.hypenate(objectClass, 'row')) .each(function(d, i){ xSpacer(d3.select(this), xValues, 0); }); @@ -4860,7 +4860,7 @@ function bubbleHeatmap( selection ) { : colorFunction$$1.dataExtent([0, Math.max(...vValues)]); cells.each(function(key, i) { - utils$1.con.log('bubbleHeatmap', 'each cell', {key: key, index: i, node: d3.select(this).node()}); + utils.con.log('bubbleHeatmap', 'each cell', {key: key, index: i, node: d3.select(this).node()}); var t = d3.select(this), currentData = data[key], @@ -4870,9 +4870,9 @@ function bubbleHeatmap( selection ) { fillColor = colorFunction$$1(key, value, i, 'fill'), // prevent duplicate computation strokeColor = colorFunction$$1(key, value, i, 'stroke'); - utils$1.con.log('bubbleHeatmap', 'radius',{radius: radius, scaled: scale(radius), extent: extent, range:scale.range()}); + utils.con.log('bubbleHeatmap', 'radius',{radius: radius, scaled: scale(radius), extent: extent, range:scale.range()}); - var c = utils$1.sel.safeSelect(t, 'circle', utils$1.str.hypenate(objectClass,'circle')); + var c = utils.sel.safeSelect(t, 'circle', utils.str.hypenate(objectClass,'circle')); c.attr('cx', xSize / 2) .attr('cy', ySize / 2 ) .attr('r', scale(radius)) @@ -4882,7 +4882,7 @@ function bubbleHeatmap( selection ) { }); - tooltip$$1.selection(cells.selectAll('circle.'+utils$1.str.hypenate(objectClass, 'circle'))) + tooltip$$1.selection(cells.selectAll('circle.'+utils.str.hypenate(objectClass, 'circle'))) .data(data); // .keys(['r', 'v']) // .header(function(d, i){return utils.str.hypenate(data[d][xKey], data[d][yKey]) }) @@ -4895,3046 +4895,4368 @@ function bubbleHeatmap( selection ) { return bhm; } +/******************************************************************************* +** ** +** ** +** BOX AND WHISKER ** +** ** +** ** +*******************************************************************************/ /** - * Creates a heatmap + * Creates a boxwhisker * - * {@link https://sumneuron.gitlab.io/d3sm/demos/heatmap-heatmap/index.html Demo} - * @constructor heatmap + * {@link https://sumneuron.gitlab.io/d3sm/demos/box-whiskers/index.html Demo} + * @constructor boxwhisker * @param {d3.selection} selection - * @namespace heatmap - * @returns {function} heatmap + * @namespace boxwhisker + * @returns {function} boxwhisker */ -function heatmap( selection ) { - var +function boxwhisker( selection ) { + var /** - * Data to plot. Assumed to be a object, where each key corresponds to a cell - * (see {@link heatmap#data}) + * Data to plot. Assumed to be a object, where each key corresponds to a box + * (see {@link boxwhisker#data}) * @param {Object} [data=undefined] - * @memberof heatmap# + * @memberof boxwhisker# * @property */ data, - /** - * Amount of horizontal space (in pixels) avaible to render the heatmap in - * (see {@link heatmap#spaceX}) + * Which direction to render the boxes in + * (see {@link boxwhisker#orient}) + * @param {number} [orient='horizontal'] + * @memberof boxwhisker# + * @property + */ + orient = 'horizontal', + /** + * Amount of horizontal space (in pixels) avaible to render the boxwhisker in + * (see {@link boxwhisker#spaceX}) * @param {number} [spaceX=undefined] - * @memberof heatmap# + * @memberof boxwhisker# * @property */ spaceX, /** - * Amount of vertical space (in pixels) avaible to render the heatmap in - * (see {@link heatmap.spaceY}) + * Amount of vertical space (in pixels) avaible to render the boxwhisker in + * (see {@link boxwhisker.spaceY}) * @param {number} [spaceY=undefined] - * @memberof heatmap# + * @memberof boxwhisker# * @property */ spaceY, - /** - * The internal key of the cell specifiying to which x axis key it belongs - * (see {@link heatmap.xKey}) - * @param {string} [xKey='x'] - * @memberof heatmap# + * Whether or not to allow boxwhisker to render elements pass the main spatial dimension + * given the orientation (see {@link boxwhisker#orient}), where {@link boxwhisker#orient}="horizontal" + * the main dimension is {@link boxwhisker#spaceX} and where {@link boxwhisker#orient}="vertical" + * the main dimension is {@link boxwhisker#spaceY} + * @param {boolean} [overflowQ=false] + * @memberof boxwhisker# * @property */ - xKey = 'x', + overflowQ = true, + /** - * The internal key of the cell specifiying to which y axis key it belongs - * (see {@link heatmap.yKey}) - * @param {string} [yKey='y'] - * @memberof heatmap# + * An array - putatively of other arrays - depicting how boxes should be arranged + * @param {Array[]} [grouping=undefined] + * @memberof boxwhisker# * @property */ - yKey = 'y', + grouping, + quartilesKey = 'quartiles', // key in object where quartiles are stored + quartilesKeys = ["0.00", "0.25", "0.50", "0.75", "1.00"], // keys in quartiles object mapping values to min, q1, q2, q3 and max + /** - * The internal key of the cell specifiying what value to use to determine the color - * (see {@link heatmap.vKey}) - * @param {string} [vKey='v'] - * @memberof heatmap# + * How to get the value of the boxwhisker + * @param {function} [valueExtractor=function(key, index) { return data[key][quartilesKey] }] + * @memberof boxwhisker# * @property */ - vKey = 'v', - + valueExtractor = function(key, index) { return data[key][quartilesKey] }, /** - * Function for extracting the the value from xKey. - * (see {@link heatmap.xExtractor}) - * @param {function} [xExtractor=function(key, i) { return data[key][xKey] }] - * @returns {string} - * @memberof heatmap# + * How to sort the boxes - if {@link boxwhisker#grouping} is not provided. + * @param {function} [sortingFunction=descending] + * @memberof boxwhisker# * @property + * default + * function(keyA, keyB) {return d3.descending( + * valueExtractor(keyA)[quartilesKeys[4]], + * valueExtractor(keyB)[quartilesKeys[4]] + * )} */ - xExtractor = function(key, i) {return data[key][xKey] }, + sortingFunction = function(keyA, keyB) {return d3.descending( + valueExtractor(keyA)[quartilesKeys[4]], + valueExtractor(keyB)[quartilesKeys[4]] + )}, /** - * Function for extracting the the value from yKey. - * (see {@link heatmap.yExtractor}) - * @param {function} [yExtractor=function(key, i) { return data[key][yKey] }] - * @returns {string} - * @memberof heatmap# + * The scale for which boxwhisker values should be transformed by + * @param {d3.scale} [scale=d3.scaleLinear] + * @memberof boxwhisker# * @property */ - yExtractor = function(key, i) { return data[key][yKey] }, - + scale = d3.scaleLinear(), /** - * Function for extracting the the value from vKey. - * (see {@link heatmap.vExtractor}) - * @param {function} [vExtractor=function(key, i) { return data[key][vKey] }] - * @returns {number} - * @memberof heatmap# + * The padding for the domain of the scale (see {@link boxwhisker#scale}) + * @param {number} [domainPadding=0.5] + * @memberof boxwhisker# * @property */ - vExtractor = function(key, i) { return data[key][vKey] }, - - + domainPadding = 0.5, /** - * Whether or not to allow heatmap to render elements pass the main spatial dimension - * given the orientation (see {@link heatmap#orient}), where {@link heatmap#orient}="bottom" or {@link heatmap#orient}="top" - * the main dimension is {@link heatmap#spaceX} and where {@link heatmap#orient}="left" or {@link heatmap#orient}="right" - * the main dimension is {@link heatmap#spaceY} - * @param {boolean} [overflowQ=false] - * @memberof heatmap# + * Default space for the spacer (percentage) of main dimension given the orientation + * (see {@link boxwhisker#orient}), where {@link boxwhisker#orient}="horizontal" + * the main dimension is {@link boxwhisker#spaceX} and where {@link boxwhisker#orient}="vertical" + * the main dimension is {@link boxwhisker#spaceY} between boxes + * @param {number} [objectSpacer=0.05] + * @memberof boxwhisker# * @property */ - overflowQ = false, - + objectSpacer = 0.05, /** - * Default space for the spacer (percentage) of main dimension given the orientation - * (see {@link heatmap#orient}), where {@link heatmap#orient}="horizontal" - * the main dimension is {@link heatmap#spaceX} and where {@link heatmap#orient}="vertical" - * the main dimension is {@link heatmap#spaceY} between bubbles - * @param {number} [objectSpacer=0.0] - * @memberof heatmap# + * The minimum size that an object can be + * @param {number} [minObjectSize=15] + * @memberof boxwhisker# * @property */ - objectSpacer = 0.0, + minObjectSize = 15, /** - * The minimum size that an object can be in the y dimension - * @param {number} [minObjectSize=50] - * @memberof heatmap# + * The maximum size that an object can be + * @param {number} [maxObjectSize=50] + * @memberof boxwhisker# * @property */ - yMinObjectSize = 50, + maxObjectSize = 50, /** - * The minimum size that an object can be in the x dimension - * @param {number} [minObjectSize=50] - * @memberof heatmap# + * Percent of box and whisker size that whiskers will be rendered + * see {@link boxwhisker#objectSize} + * @param {number} [whiskerWidthPercent=0.6] + * @memberof boxwhisker# * @property */ - xMinObjectSize = 50, + whiskerWidthPercent = .6, /** - * The maximum size that an object can be in the x dimension - * @param {number} [maxObjectSize=100] - * @memberof heatmap# + * Instance of ColorFunction + * @param {function} [colorFunction = colorFunction()] + * @memberof boxwhisker# * @property */ - xMaxObjectSize = 100, + colorFunction$$1 = colorFunction(), /** - * The maximum size that an object can be in the y dimension - * @param {number} [maxObjectSize=100] - * @memberof heatmap# + * The stroke width of the boxes + * @param {number} [boxStrokeWidth=2] + * @memberof boxwhisker# * @property */ - yMaxObjectSize = 100, - - + boxStrokeWidth = 2, /** - * The stroke width of the bubbles - * @param {number} [objectStrokeWidth=2] - * @memberof heatmap# + * The stroke width of the whiskers + * @param {number} [whiskerStrokeWidth=2] + * @memberof boxwhisker# * @property */ - objectStrokeWidth = 2, - // colorFunc = colorFunction(), + whiskerStrokeWidth = 2, /** * Color of the background * @param {string} [backgroundFill="transparent"] - * @memberof heatmap# + * @memberof boxwhisker# * @property */ backgroundFill = 'transparent', /** - * Namespace for all items made by this instance of heatmap - * @param {string} [namespace="d3sm-heatmap"] - * @memberof heatmap# + * Namespace for all items made by this instance of boxwhisker + * @param {string} [namespace="d3sm-box-whisker"] + * @memberof boxwhisker# * @property */ - namespace = 'd3sm-heatmap', + namespace = 'd3sm-box-whisker', /** - * Class name for heatmap container ( element) - * @param {string} [objectClass="heatmap"] - * @memberof heatmap# + * Class name for boxwhisker container ( element) + * @param {string} [objectClass="box-whisk"] + * @memberof boxwhisker# * @property */ - objectClass = 'heatmap', + objectClass = 'box-whisk', + /** * Duration of all transitions of this element * @param {number} [transitionDuration=1000] - * @memberof heatmap# + * @memberof boxwhisker# * @property */ transitionDuration = 1000, /** * Easing function for transitions * @param {d3.ease} [easeFunc=d3.easeExp] - * @memberof heatmap# + * @memberof boxwhisker# * @property */ easeFunc = d3.easeExp, /** - * Stores the keys of all the cells - * Calculated after heatmap called. - * @param {string[]} [cellKeys=undefined] - * @memberof heatmap# - * @property - */ - cellKeys, - /** - * Stores the list of utils.arr.unique xValues - * Calculated after heatmap called. - * @param {string[]} [xValues=undefined] - * @memberof heatmap# + * The keys of the boxes + * @param {string[]} [boxKeys=undefined] + * @memberof boxwhisker# * @property */ - xValues, + boxKeys, /** - * Stores the list of utils.arr.unique yValues - * Calculated after heatmap called. - * @param {string[]} [yValues=undefined] - * @memberof heatmap# + * The values of the boxes + * @param {string[]} [boxValues=undefined] + * @memberof boxwhisker# * @property */ - yValues, + boxValues, /** - * Stores the list of utils.arr.unique vValues - * Calculated after heatmap called. - * @param {string[]} [vValues=undefined] - * @memberof heatmap# + * The objectSize (actual width) used by the boxes + * @param {number} [objectSize=undefined] + * @memberof boxwhisker# * @property */ - vValues, - - xKeySortingFunction = function(a, b) { return xValues.indexOf(xExtractor(a)) - xValues.indexOf(xExtractor(b)) }, - yKeySortingFunction = function(a, b) { return yValues.indexOf(yExtractor(a)) - yValues.indexOf(yExtractor(b)) }, + objectSize, /** - * Instance of ColorFunction with .colorBy set to 'category' - * @function colorFunction - * @memberof heatmap# + * The spacerSize (actual width) used by the spacers between the boxes + * @param {number} [spacerSize=undefined] + * @memberof boxwhisker# * @property */ - colorFunction$$1 = colorFunction().colorBy('category'), + spacerSize, /** * Instance of Tooltip - * @function tooltip - * @memberof heatmap# - * @property - */ - tooltip$$1 = tooltip(), - - /** - * store the size the heatmap could be in the x dimension - * the actuall size of the heatmap will be the min of xSize and {@link heatmap#ySize} - * Calculated after heatmap called. - * @param {string[]} [xSize=undefined] - * @memberof heatmap# - * @property - */ - xSize, - /** - * store the size of the spacer in the x dimension - * Calculated after heatmap called. - * @param {string[]} [xSpacerSize=undefined] - * @memberof heatmap# - * @property - */ - xSpacerSize, - - /** - * store the size the heatmap could be in the y dimension - * the actuall size of the heatmap will be the min of xSize and {@link heatmap#xSize} - * Calculated after heatmap called. - * @param {string[]} [ySize=undefined] - * @memberof heatmap# - * @property - */ - ySize, - /** - * store the size of the spacer in the y dimension. - * Calculated after heatmap called. - * @param {string[]} [xSpacerSize=undefined] - * @memberof heatmap# + * @param {function} [tooltip=tooltip()] + * @memberof boxwhisker# * @property */ - ySpacerSize; - + tooltip$$1 = tooltip(); /** * Gets or sets the selection in which items are manipulated * @param {d3.selection} [_=none] - * @returns {heatmap | d3.selection} - * @memberof heatmap + * @returns {boxwhisker | d3.selection} + * @memberof boxwhisker * @property * by default selection = selection */ - hm.selection = function(_) { return arguments.length ? (selection = _, hm) : selection; }; + boxwhisker.selection = function(_) { return arguments.length ? (selection = _, boxwhisker) : selection; }; /** * Gets or sets the data - * (see {@link heatmap#data}) + * (see {@link boxwhisker#data}) * @param {number} [_=none] - * @returns {heatmap | object} - * @memberof heatmap + * @returns {boxwhisker | object} + * @memberof boxwhisker * @property */ - hm.data = function(_) { return arguments.length ? (data = _, hm) : data; }; - // hm.orient = function(_) { return arguments.length ? (orient = _, hm) : orient; } + boxwhisker.data = function(_) { return arguments.length ? (data = _, boxwhisker) : data; }; + /** + * Gets or sets the orient of the boxes + * (see {@link boxwhisker#orient}) + * @param {number} [_=none] + * @returns {boxwhisker | object} + * @memberof boxwhisker + * @property + */ + boxwhisker.orient = function(_) { return arguments.length ? (orient = _, boxwhisker) : orient; }; /** * Gets or sets the amount of horizontal space in which items are manipulated - * (see {@link heatmap#spaceX}) + * (see {@link boxwhisker#spaceX}) * @param {number} [_=none] should be a number > 0 - * @returns {heatmap | number} - * @memberof heatmap + * @returns {boxwhisker | number} + * @memberof boxwhisker * @property * by default spaceX = undefined */ - hm.spaceX = function(_) { return arguments.length ? (spaceX = _, hm) : spaceX; }; + boxwhisker.spaceX = function(_) { return arguments.length ? (spaceX = _, boxwhisker) : spaceX; }; /** * Gets or sets the amount of vertical space in which items are manipulated - * (see {@link heatmap#spaceY}) + * (see {@link boxwhisker#spaceY}) * @param {number} [_=none] should be a number > 0 - * @returns {heatmap | number} - * @memberof heatmap + * @returns {boxwhisker | number} + * @memberof boxwhisker * @property * by default spaceY = undefined */ - hm.spaceY = function(_) { return arguments.length ? (spaceY = _, hm) : spaceY; }; - + boxwhisker.spaceY = function(_) { return arguments.length ? (spaceY = _, boxwhisker) : spaceY; }; /** - * Gets or sets the xKey - * (see {@link heatmap#xKey}) - * @param {string} [_=none] - * @returns {heatmap | string} - * @memberof heatmap + * Gets / sets whether or not boxwhisker is allowed to go beyond specified dimensions + * (see {@link boxwhisker#overflowQ}) + * @param {boolean} [_=none] + * @returns {boxwhisker | boolean} + * @memberof boxwhisker * @property - * by default xKey = 'x' + * by default overflowQ = false */ - hm.xKey = function(_) { return arguments.length ? (xKey = _, hm) : xKey; }; + boxwhisker.overflowQ = function(_) { return arguments.length ? (overflowQ = _, boxwhisker) : overflowQ; }; /** - * Gets or sets the yKey - * (see {@link heatmap#yKey}) - * @param {string} [_=none] - * @returns {heatmap | string} - * @memberof heatmap + * Gets / sets the grouping of the boxes + * (see {@link boxwhisker#grouping}) + * @param {Array[]} [_=none] + * @returns {boxwhisker | Array[]} + * @memberof boxwhisker * @property - * by default yKey = 'y' + * by default grouping = undefined */ - hm.yKey = function(_) { return arguments.length ? (yKey = _, hm) : yKey; }; - + boxwhisker.grouping = function(_) { return arguments.length ? (grouping = _, boxwhisker) : grouping; }; /** - * Gets or sets the vKey - * (see {@link heatmap#vKey}) + * Gets / sets the quartilesKey + * (see {@link boxwhisker#quartilesKey}) * @param {string} [_=none] - * @returns {heatmap | string} - * @memberof heatmap + * @returns {boxwhisker | string} + * @memberof boxwhisker * @property - * by default vKey = 'y' + * by default quartilesKey = quartiles */ - hm.vKey = function(_) { return arguments.length ? (vKey = _, hm) : vKey; }; - + boxwhisker.quartilesKey = function(_) { return arguments.length ? (quartilesKey = _, boxwhisker) : quartilesKey; }; /** - * Gets or sets the cellKeys - * (see {@link heatmap#cellKeys}) + * Gets / sets the quartilesKeys + * (see {@link boxwhisker#quartilesKeys}) * @param {string[]} [_=none] - * @returns {heatmap | string[]} - * @memberof heatmap + * @returns {boxwhisker | string[]} + * @memberof boxwhisker * @property - * by default cellKeys = undefined + * by default quartilesKeys = [Q0, Q1, Q2, Q3, Q4] */ - hm.cellKeys = function(_) { return arguments.length ? (cellKeys = _, hm) : cellKeys; }; + boxwhisker.quartilesKeys = function(_) { return arguments.length ? (quartilesKeys = _, boxwhisker) : quartilesKeys; }; /** - * Gets or sets the xValues - * (see {@link heatmap#xValues}) - * @param {string[]} [_=none] - * @returns {heatmap | string[]} - * @memberof heatmap + * Gets / sets the valueExtractor + * (see {@link boxwhisker#valueExtractor}) + * @param {function} [_=none] + * @returns {boxwhisker | function} + * @memberof boxwhisker * @property - * by default xValues = undefined + * by default valueExtractor = function(key, index) { return data[key][quartilesKey] }, */ - hm.xValues = function(_) { return arguments.length ? (xValues = _, hm) : xValues; }; + + boxwhisker.valueExtractor = function(_) { return arguments.length ? (valueExtractor = _, boxwhisker) : valueExtractor; }; /** - * Gets or sets the yValues - * (see {@link heatmap#yValues}) - * @param {string[]} [_=none] - * @returns {heatmap | string[]} - * @memberof heatmap + * Gets / sets the sortingFunction + * (see {@link boxwhisker#sortingFunction}) + * @param {function} [_=none] + * @returns {boxwhisker | function} + * @memberof boxwhisker * @property - * by default yValues = undefined + * by default sortingFunction = function(keyA, keyB) {return d3.descending( + * valueExtractor(keyA)[quartilesKeys[4]], + * valueExtractor(keyB)[quartilesKeys[4]] + * )}, */ - hm.yValues = function(_) { return arguments.length ? (yValues = _, hm) : yValues; }; + boxwhisker.sortingFunction = function(_) { return arguments.length ? (sortingFunction = _, boxwhisker) : sortingFunction; }; /** - * Gets or sets the vValues - * (see {@link heatmap#vValues}) - * @param {number[]} [_=none] - * @returns {heatmap | number[]} - * @memberof heatmap + * Gets / sets the scale for which the boxwhisker values should be transformed by + * (see {@link boxwhisker#scale}) + * @param {d3.scale} [_=none] + * @returns {boxwhisker | d3.scale} + * @memberof boxwhisker * @property - * by default vValues = undefined + * by default scale = d3.scaleLinear() */ - hm.vValues = function(_) { return arguments.length ? (vValues = _, hm) : vValues; }; - - + boxwhisker.scale = function(_) { return arguments.length ? (scale = _, boxwhisker) : scale; }; /** - * Gets or sets the xExtractor - * (see {@link heatmap#xExtractor}) - * @param {function} [_=none] - * @returns {heatmap | function} - * @memberof heatmap + * Gets / sets the padding for the domain of the scale + * (see {@link boxwhisker#domainPadding}) + * @param {number} [_=none] + * @returns {boxwhisker | number} + * @memberof boxwhisker * @property - * by default xExtractor = undefined + * by default domainPadding = 0.5 */ - hm.xExtractor = function(_) { return arguments.length ? (xExtractor = _, hm) : xExtractor; }; + boxwhisker.domainPadding = function(_) { return arguments.length ? (domainPadding = _, boxwhisker) : domainPadding; }; /** - * Gets or sets the yExtractor - * (see {@link heatmap#yExtractor}) - * @param {function} [_=none] - * @returns {heatmap | function} - * @memberof heatmap + * Gets / sets objectSpacer + * (see {@link boxwhisker#objectSpacer}) + * @param {number} [_=none] + * @returns {boxwhisker | number} + * @memberof boxwhisker * @property - * by default yExtractor = undefined + * by default objectSpacer = 0.05 */ - hm.yExtractor = function(_) { return arguments.length ? (yExtractor = _, hm) : yExtractor; }; + boxwhisker.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, boxwhisker) : objectSpacer; }; /** - * Gets or sets the vExtractor - * (see {@link heatmap#vExtractor}) - * @param {function} [_=none] - * @returns {heatmap | function} - * @memberof heatmap + * Gets / sets the minObjectSize + * (see {@link boxwhisker#minObjectSize}) + * @param {number} [_=none] + * @returns {boxwhisker | number} + * @memberof boxwhisker * @property - * by default vExtractor = undefined + * by default minObjectSize = 15 */ - hm.vExtractor = function(_) { return arguments.length ? (vExtractor = _, hm) : vExtractor; }; - - /** - * Gets / sets whether or not heatmap is allowed to go beyond specified dimensions - * (see {@link heatmap#spaceX}) - * @param {boolean} [_=none] - * @returns {heatmap | boolean} - * @memberof heatmap - * @property - * by default overflowQ = false - */ - hm.overflowQ = function(_) { return arguments.length ? (overflowQ = _, hm) : overflowQ; }; - /** - * Gets / sets objectSpacer - * (see {@link heatmap#objectSpacer}) - * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap - * @property - * by default objectSpacer = 0.0 - */ - hm.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, objectSpacer) : data; }; + boxwhisker.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, boxwhisker) : minObjectSize; }; /** - * Gets / sets the minObjectSize - * (see {@link heatmap#minObjectSize}) + * Gets / sets the maxObjectSize + * (see {@link boxwhisker#maxObjectSize}) * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * @returns {boxwhisker | number} + * @memberof boxwhisker * @property - * by default minObjectSize = 50 + * by default maxObjectSize = 50 */ - hm.yMinObjectSize = function(_) { return arguments.length ? (yMinObjectSize = _, hm) : yMinObjectSize; }; + boxwhisker.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, boxwhisker) : maxObjectSize; }; /** - * Gets / sets the maxObjectSize - * (see {@link heatmap#maxObjectSize}) + * Gets / sets the whiskerWidthPercent + * (see {@link boxwhisker#whiskerWidthPercent}) * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * @returns {boxwhisker | number} + * @memberof boxwhisker * @property - * by default maxObjectSize = 100 + * by default maxObjectSize = 0.6 */ - hm.yMaxObjectSize = function(_) { return arguments.length ? (yMaxObjectSize = _, hm) : yMaxObjectSize; }; + boxwhisker.whiskerWidthPercent = function(_) { return arguments.length ? (whiskerWidthPercent = _, boxwhisker) : whiskerWidthPercent; }; /** - * Gets / sets the minObjectSize - * (see {@link heatmap#minObjectSize}) - * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * Gets / sets the colorFunction + * (see {@link boxwhisker#colorFunction}) + * @param {colorFunction} [_=none] + * @returns {boxwhisker | colorFunction} + * @memberof boxwhisker * @property - * by default minObjectSize = 50 + * by default colorFunction = colorFunction() */ - hm.xMinObjectSize = function(_) { return arguments.length ? (xMinObjectSize = _, hm) : xMinObjectSize; }; + boxwhisker.colorFunction = function(_) { return arguments.length ? (colorFunction$$1 = _, boxwhisker) : colorFunction$$1; }; /** - * Gets / sets the maxObjectSize - * (see {@link heatmap#maxObjectSize}) + * Gets / sets the boxStrokeWidth + * (see {@link boxwhisker#boxStrokeWidth}) * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * @returns {boxwhisker | number} + * @memberof boxwhisker * @property - * by default maxObjectSize = 100 + * by default boxStrokeWidth = 2 */ - hm.xMaxObjectSize = function(_) { return arguments.length ? (xMaxObjectSize = _, hm) : xMaxObjectSize; }; + boxwhisker.boxStrokeWidth = function(_) { return arguments.length ? (boxStrokeWidth = _, boxwhisker) : boxStrokeWidth; }; /** - * Gets / sets the objectStrokeWidth - * (see {@link heatmap#objectStrokeWidth}) + * Gets / sets the whiskerStrokeWidth + * (see {@link boxwhisker#whiskerStrokeWidth}) * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * @returns {boxwhisker | number} + * @memberof boxwhisker * @property - * by default objectStrokeWidth = 2 + * by default whiskerStrokeWidth = 2 */ - hm.objectStrokeWidth = function(_) { return arguments.length ? (objectStrokeWidth = _, hm) : objectStrokeWidth; }; + boxwhisker.whiskerStrokeWidth = function(_) { return arguments.length ? (whiskerStrokeWidth = _, boxwhisker) : whiskerStrokeWidth; }; + /** * Gets / sets the backgroundFill - * (see {@link heatmap#backgroundFill}) + * (see {@link boxwhisker#backgroundFill}) * @param {string} [_=none] - * @returns {heatmap | string} - * @memberof heatmap + * @returns {boxwhisker | string} + * @memberof boxwhisker * @property * by default backgroundFill = 'transparent' */ - hm.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, hm) : backgroundFill; }; + boxwhisker.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, boxwhisker) : backgroundFill; }; /** * Gets / sets the namespace - * (see {@link heatmap#namespace}) + * (see {@link boxwhisker#namespace}) * @param {string} [_=none] - * @returns {heatmap | string} - * @memberof heatmap + * @returns {boxwhisker | string} + * @memberof boxwhisker * @property - * by default namespace = 'd3sm-heatmap' + * by default namespace = 'd3sm-boxwhisker' */ - hm.namespace = function(_) { return arguments.length ? (namespace = _, hm) : namespace; }; + boxwhisker.namespace = function(_) { return arguments.length ? (namespace = _, boxwhisker) : namespace; }; /** * Gets / sets the objectClass - * (see {@link heatmap#objectClass}) + * (see {@link boxwhisker#objectClass}) * @param {string} [_=none] - * @returns {heatmap | string} - * @memberof heatmap + * @returns {boxwhisker | string} + * @memberof boxwhisker * @property * by default objectClass = 'tick-group' */ - hm.objectClass = function(_) { return arguments.length ? (objectClass = _, hm) : objectClass; }; + boxwhisker.objectClass = function(_) { return arguments.length ? (objectClass = _, boxwhisker) : objectClass; }; /** * Gets / sets the transitionDuration - * (see {@link heatmap#transitionDuration}) + * (see {@link boxwhisker#transitionDuration}) * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * @returns {boxwhisker | number} + * @memberof boxwhisker * @property * by default transitionDuration = 1000 */ - hm.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, hm) : transitionDuration; }; + boxwhisker.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, boxwhisker) : transitionDuration; }; /** * Gets / sets the easeFunc - * (see {@link heatmap#easeFunc}) + * (see {@link boxwhisker#easeFunc}) * @param {d3.ease} [_=none] - * @returns {heatmap | d3.ease} - * @memberof heatmap + * @returns {boxwhisker | d3.ease} + * @memberof boxwhisker * @property * by default easeFunc = d3.easeExp */ - hm.easeFunc = function(_) { return arguments.length ? (easeFunc = _, hm) : easeFunc; }; - - /** - * Gets / sets the tooltip - * (see {@link heatmap#tooltip}) - * @param {tooltip} [_=none] - * @returns {heatmap | tooltip} - * @memberof heatmap - * @property - * by default tooltip = tooltip() - */ - hm.tooltip = function(_) { return arguments.length ? (tooltip$$1 = _, hm) : tooltip$$1; }; + boxwhisker.easeFunc = function(_) { return arguments.length ? (easeFunc = _, boxwhisker) : easeFunc; }; /** - * Gets / sets the colorFunction - * (see {@link heatmap#colorFunction}) - * @param {colorFunction} [_=none] - * @returns {heatmap | colorFunction} - * @memberof heatmap + * Gets / sets the barKeys + * (see {@link boxwhisker#boxKeys}) + * @param {string[]} [_=none] + * @returns {boxwhisker | string[]} + * @memberof boxwhisker * @property - * by default colorFunction = colorFunction() + * by default boxKeys = undefined */ - hm.colorFunction = function(_) { return arguments.length ? (colorFunction$$1 = _, hm) : colorFunction$$1; }; - + boxwhisker.boxKeys = function(_) { return arguments.length ? (boxKeys = _, boxwhisker) : boxKeys; }; /** - * Gets / sets the xSize - * (see {@link heatmap#xSize}) - * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * Gets / sets the boxValues + * (see {@link boxwhisker#boxValues}) + * @param {number[]} [_=none] + * @returns {boxwhisker | number[]} + * @memberof boxwhisker * @property - * by default xSize = undefined + * by default boxValues = undefined */ - hm.xSize = function(_) { return arguments.length ? (xSize = _, hm) : xSize; }; + boxwhisker.boxValues = function(_) { return arguments.length ? (boxValues = _, boxwhisker) : boxValues; }; /** - * Gets / sets the xSpacerSize - * (see {@link heatmap#xSpacerSize}) + * Gets / sets the objectSize + * (see {@link boxwhisker#objectSize}) * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * @returns {boxwhisker | number} + * @memberof boxwhisker * @property - * by default xSpacerSize = undefined + * by default objectSize = undefined */ - hm.xSpacerSize = function(_) { return arguments.length ? (xSpacerSize = _, hm) : xSpacerSize; }; + boxwhisker.objectSize = function(_) { return arguments.length ? (objectSize = _, boxwhisker) : objectSize; }; /** - * Gets / sets the ySize - * (see {@link heatmap#ySize}) + * Gets / sets the spacerSize + * (see {@link boxwhisker#spacerSize}) * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * @returns {boxwhisker | number} + * @memberof boxwhisker * @property - * by default ySize = undefined + * by default spacerSize = undefined */ - hm.ySize = function(_) { return arguments.length ? (ySize = _, hm) : ySize; }; + boxwhisker.spacerSize = function(_) { return arguments.length ? (spacerSize = _, boxwhisker) : spacerSize; }; /** - * Gets / sets the ySpacerSize - * (see {@link heatmap#ySpacerSize}) - * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * Gets / sets the tooltip + * (see {@link boxwhisker#tooltip}) + * @param {tooltip} [_=none] + * @returns {boxwhisker | tooltip} + * @memberof boxwhisker * @property - * by default ySpacerSize = undefined + * by default tooltip = tooltip() */ - hm.ySpacerSize = function(_) { return arguments.length ? (ySpacerSize = _, hm) : ySpacerSize; }; - // hm.yKeySortingFunction = function(_) { return arguments.length ? (yKeySortingFunction = _, hm) : yKeySortingFunction; } - // hm.xKeySortingFunction = function(_) { return arguments.length ? (xKeySortingFunction = _, hm) : xKeySortingFunction; } + boxwhisker.tooltip = function(_) { return arguments.length ? (tooltip$$1 = _, boxwhisker) : tooltip$$1; }; + function boxwhisker() { + // for convenience in handling orientation specific values + var horizontalQ = (orient == 'horizontal') ? true : false; + var verticalQ = !horizontalQ; - function hm() { + // background cliping rectangle var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; - var container = utils$1.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); - - cellKeys = d3.keys(data); + var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); - xValues = utils$1.arr.unique(cellKeys.map(xExtractor)); - yValues = utils$1.arr.unique(cellKeys.map(yExtractor)); - vValues = utils$1.arr.unique(cellKeys.map(vExtractor)); + // if grouping is undefined sort keys by sorting funct + var ordered = (grouping == undefined) ? d3.keys(data).sort(sortingFunction) : grouping; + // to prevent re-calculation and getters to be passed to axes + boxKeys = utils.arr.flatten(ordered); + boxValues = boxKeys.map(valueExtractor); - cellKeys.sort(function(a, b){ return xKeySortingFunction(a, b) || yKeySortingFunction(a, b) }); - utils$1.con.log('heatmap', 'cells are sorted by', cellKeys); + var numberOfObjects = boxKeys.length; + var extent = [ + Math.min(...boxValues.map(function(d,i){return d[quartilesKeys[0]]})) - domainPadding, + Math.max(...boxValues.map(function(d,i){return d[quartilesKeys[4]]})) + domainPadding + ]; + // set the scale + scale.domain(extent).range(horizontalQ ? [0,spaceY] : [spaceX, 0]); + var space = horizontalQ ? spaceX : spaceY; + // calculate object size + objectSize = utils.math.calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ); + // calculate spacer size if needed + spacerSize = utils.math.calculateWidthOfSpacer(boxKeys, space, objectSize, numberOfObjects, objectSpacer, overflowQ); + // make the nested groups + var spacerFunction = groupingSpacer() + .horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects) + .objectClass(objectClass).objectSize(objectSize).spacerSize(spacerSize) + .transitionDuration(transitionDuration).easeFunc(easeFunc) + .namespace(namespace); - utils$1.con.log('heatmap', 'x and y keys are', {x: xValues, y:yValues}); + // move stuff + spacerFunction(container, ordered, 0); + var parentIndexArray = []; + container.selectAll('g:not(.to-remove).'+objectClass) + .each(function(d, i){if (utils.arr.hasQ(boxKeys, d)){ parentIndexArray.push(Number(d3.select(this).attr('parent-index')));}}); - var xDim = xValues.length, yDim = yValues.length; - ySize = utils$1.math.calculateWidthOfObject(spaceY, yDim, yMinObjectSize, yMaxObjectSize, objectSpacer, overflowQ); - xSize = utils$1.math.calculateWidthOfObject(spaceX, xDim, xMinObjectSize, xMaxObjectSize, objectSpacer, overflowQ); - ySpacerSize = utils$1.math.calculateWidthOfSpacer(yValues, spaceY, ySize, yDim, objectSpacer, overflowQ); - xSpacerSize = utils$1.math.calculateWidthOfSpacer(xValues, spaceX, xSize, xDim, objectSpacer, overflowQ); - // console.table({ - // x:{ - // object: xSize, - // spacer: xSpacerSize, - // dim: xDim - // }, - // y:{ - // object: ySize, - // spacer: ySpacerSize, - // dim: yDim - // } - // - // }) - utils$1.con.log('heatmap', 'size of', {x: xSize, y: ySize}); + colorFunction$$1 = colorFunction$$1.colorBy() == 'index' + ? colorFunction$$1.dataExtent([0, Math.max(...parentIndexArray)]) + : colorFunction$$1.dataExtent(extent); + // set attributes for box and whiskers + container.selectAll('g:not(.to-remove).'+objectClass).each(function(key, i) { + var t = d3.select(this), + currentData = data[key], - var ySpacer = groupingSpacer() - .horizontalQ(false) - .moveby('category') - .numberOfObjects(yDim) - .objectClass(utils$1.str.hypenate(objectClass, 'row')) - .objectSize(ySize + ySpacerSize) - .spacerSize(0) - .transitionDuration(transitionDuration) - .easeFunc(easeFunc) - .namespace('row'); + quartiles$$1 = valueExtractor(key, i), + q0 = quartiles$$1[quartilesKeys[0]], + q1 = quartiles$$1[quartilesKeys[1]], + q2 = quartiles$$1[quartilesKeys[2]], + q3 = quartiles$$1[quartilesKeys[3]], + q4 = quartiles$$1[quartilesKeys[4]]; - var xSpacer = groupingSpacer() - .horizontalQ(true) - .moveby('category') - .numberOfObjects(xDim) - .objectClass(objectClass) - .objectSize(xSize + xSpacerSize) - .spacerSize(0) - .transitionDuration(transitionDuration) - .easeFunc(easeFunc); + var i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'), + fillColor = colorFunction$$1(key, q2, i, 'fill'), // prevent duplicate computation + strokeColor = colorFunction$$1(key, q2, i, 'stroke'); - ySpacer(container, yValues, 0); - container.selectAll('g.'+utils$1.str.hypenate(objectClass, 'row')) - .each(function(d, i){ xSpacer(d3.select(this), xValues, 0); }); + var + whisk = utils.sel.safeSelect(t, 'g', 'whisker'), + uWhisk = utils.sel.safeSelect(whisk, 'path', 'upper'), + lWhisk = utils.sel.safeSelect(whisk, 'path', 'lower'), + quart = utils.sel.safeSelect(t, 'g', 'quartile'), + uQuart = utils.sel.safeSelect(quart, 'rect', 'upper'), + lQuart = utils.sel.safeSelect(quart, 'rect', 'lower'), + mQuart = utils.sel.safeSelect(quart, 'circle', 'median'); + + + // set upper quartile (q3) + uQuart.transition().duration(transitionDuration).ease(easeFunc) + .attr('width', horizontalQ ? objectSize : scale(q3) - scale(q2)) + .attr('height', verticalQ ? objectSize : scale(q3) - scale(q2)) + .attr('fill', fillColor) + .attr('stroke', strokeColor) + .attr('stroke-width', boxStrokeWidth) + .attr('transform', function(d, i){ + var + x = horizontalQ ? 0 : scale(q2), + y = verticalQ ? 0 : scale(extent[1]) - scale(q3), + t = 'translate('+x+','+y+')'; + return t + }); - var cells = container.selectAll('g:not(.to-remove).'+objectClass); + // set lower quartile (q1) + lQuart.transition().duration(transitionDuration).ease(easeFunc) + .attr('width', horizontalQ ? objectSize : scale(q2) - scale(q1)) + .attr('height', verticalQ ? objectSize : scale(q2) - scale(q1)) + .attr('fill', fillColor) + .attr('stroke', strokeColor) + .attr('stroke-width', boxStrokeWidth) + .attr('transform', function(d, i){ + var + x = horizontalQ ? 0 : scale(q1), + y = verticalQ ? 0 : scale(extent[1]) - scale(q2), + t = 'translate('+x+','+y+')'; + return t + }); - if (cellKeys.length != yValues.length * xValues.length) { - var lookup = {}; - cellKeys.map(function(k, i){ - lookup[xExtractor(k)+'::'+yExtractor(k)] = k; + // set median (q2) + mQuart.transition().duration(transitionDuration).ease(easeFunc) + .attr('r', function(d, i){ + var r = objectSize / 2; + var dif = (scale(q3) - scale(q1)) / 2; + return (r > dif) ? dif : r + }) + .attr('fill', fillColor) + .attr('stroke', strokeColor) + .attr('stroke-width', boxStrokeWidth) + .attr('transform', function(d, i){ + var + x = horizontalQ ? objectSize / 2 : scale(q2), + y = verticalQ ? objectSize / 2 : scale(extent[1]) - scale(q2), + t = 'translate('+x+','+y+')'; + return t }); - var positionedCellKeys = []; - for (var i = 0; i < yValues.length; i++) { - for (var j = 0; j < xValues.length; j++) { - var lookupValue = lookup[xValues[j]+"::"+yValues[i]]; - if (lookupValue == undefined) { - positionedCellKeys.push(undefined); - } else { - positionedCellKeys.push(lookupValue); - } - } - } + // set lower whisker (min) + lWhisk.transition().duration(transitionDuration).ease(easeFunc) + .attr('d', function(dd, ii){ + var + dir = false, + x = 0, + y = 0, + h = horizontalQ ? scale(q1) - scale(q0) : objectSize, + w = verticalQ ? scale(q1) - scale(q0) : objectSize; + return utils.paths.whiskerPath(dir, x, y, w, h, whiskerWidthPercent, orient) + }) + .attr('transform', function(d, i){ + var + x = horizontalQ ? 0 : scale(q1), + y = verticalQ ? 0 : scale(extent[1]) - scale(q1), + t = 'translate('+x+','+y+')'; + return t + }) + .attr('stroke', 'black').attr('stroke-width', whiskerStrokeWidth) + .attr('fill', 'none'); - cells.data(positionedCellKeys); - - // maybe breaks this - // !!!!!! IMPORTANT NOTE TODO LOOK HERE BUG - cellKeys = positionedCellKeys; - } else { - cells.data(cellKeys); - } - - - - var parentIndexArray = []; - cells.each(function(d, i){ parentIndexArray.push(Number(d3.select(this).attr('parent-index'))); }); - - colorFunction$$1 = colorFunction$$1.colorBy() == 'index' - ? colorFunction$$1.dataExtent([0, Math.max(...parentIndexArray)]) - : colorFunction$$1.dataExtent([0, Math.max(...vValues)]); - - cells.each(function(key, i) { - utils$1.con.log('heatmap', 'each cell', {key: key, index: i, node: d3.select(this).node()}); - - var t = d3.select(this); - if (key == undefined) {return} - var currentData = data[key], - value = vExtractor(key, i), - i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'), - fillColor = colorFunction$$1(key, value, i, 'fill'), // prevent duplicate computation - strokeColor = colorFunction$$1(key, value, i, 'stroke'); - - var c = utils$1.sel.safeSelect(t, 'rect', utils$1.str.hypenate(objectClass,'rect')); - c.attr('width', xSize + xSpacerSize - objectStrokeWidth) - .attr('height', ySize + ySpacerSize - objectStrokeWidth) - .attr('fill', fillColor) - .attr('x', objectStrokeWidth/2) - .attr('y', objectStrokeWidth/2) - .attr('stroke', "#000") - .attr('stroke-width', objectStrokeWidth); + // set upper whisker (max) + uWhisk.transition().duration(transitionDuration).ease(easeFunc) + .attr('d', function(dd, ii){ + var + dir = true, + x = 0, + y = 0, + h = horizontalQ ? scale(q4) - scale(q3) : objectSize, + w = verticalQ ? scale(q4) - scale(q3) : objectSize; + return utils.paths.whiskerPath(dir, x, y, w, h, whiskerWidthPercent, orient) + }) + .attr('transform', function(d, i){ + var + x = horizontalQ ? 0 : scale(q3), + y = verticalQ ? 0 : scale(extent[1]) - scale(q4), + t = 'translate('+x+','+y+')'; + return t + }) + .attr('stroke', 'black') + .attr('stroke-width', whiskerStrokeWidth) + .attr('fill', 'none'); }); - tooltip$$1.selection(cells.selectAll('rect.'+utils$1.str.hypenate(objectClass, 'rect'))) + tooltip$$1.selection(container.selectAll('g:not(.to-remove).'+objectClass)) .data(data); - // .keys(['r', 'v']) - // .header(function(d, i){return utils.str.hypenate(data[d][xKey], data[d][yKey]) }) - tooltip$$1(); } - return hm; + return boxwhisker } -/******************************************************************************* -** ** -** ** -** VIOLIN ** -** ** -** ** -*******************************************************************************/ - /** - * Creates a violin + * Creates a heatmap * - * {@link https://sumneuron.gitlab.io/d3sm/demos/basic-violins/index.html Demo} - * @constructor violin + * {@link https://sumneuron.gitlab.io/d3sm/demos/heatmap-heatmap/index.html Demo} + * @constructor heatmap * @param {d3.selection} selection - * @namespace violin - * @returns {function} violin + * @namespace heatmap + * @returns {function} heatmap */ -function violin( selection ) { +function heatmap( selection ) { var /** - * Data to plot. Assumed to be a object, where each key corresponds to a violin - * (see {@link violin#data}) + * Data to plot. Assumed to be a object, where each key corresponds to a cell + * (see {@link heatmap#data}) * @param {Object} [data=undefined] - * @memberof violin# + * @memberof heatmap# * @property */ data, + /** - * Which direction to render the bars in - * (see {@link violin#orient}) - * @param {number} [orient='horizontal'] - * @memberof violin# - * @property - */ - orient='horizontal', - /** - * Amount of horizontal space (in pixels) avaible to render the violin in - * (see {@link violin#spaceX}) + * Amount of horizontal space (in pixels) avaible to render the heatmap in + * (see {@link heatmap#spaceX}) * @param {number} [spaceX=undefined] - * @memberof violin# + * @memberof heatmap# * @property */ spaceX, /** - * Amount of vertical space (in pixels) avaible to render the violin in - * (see {@link violin.spaceY}) + * Amount of vertical space (in pixels) avaible to render the heatmap in + * (see {@link heatmap.spaceY}) * @param {number} [spaceY=undefined] - * @memberof violin# + * @memberof heatmap# * @property */ spaceY, + /** - * Whether or not to allow violin to render elements pass the main spatial dimension - * given the orientation (see {@link violin#orient}), where {@link violin#orient}="horizontal" - * the main dimension is {@link violin#spaceX} and where {@link violin#orient}="vertical" - * the main dimension is {@link violin#spaceY} - * @param {boolean} [overflowQ=false] - * @memberof violin# + * The internal key of the cell specifiying to which x axis key it belongs + * (see {@link heatmap.xKey}) + * @param {string} [xKey='x'] + * @memberof heatmap# * @property */ - overflowQ = true, + xKey = 'x', /** - * Whether or not to display points inside the points - * @param {boolean} [pointsQ=false] - * @memberof violin# + * The internal key of the cell specifiying to which y axis key it belongs + * (see {@link heatmap.yKey}) + * @param {string} [yKey='y'] + * @memberof heatmap# * @property */ - pointsQ = true, + yKey = 'y', + /** - * An array - putatively of other arrays - depicting how bars should be arranged - * @param {Array[]} [grouping=undefined] - * @memberof violin# + * The internal key of the cell specifiying what value to use to determine the color + * (see {@link heatmap.vKey}) + * @param {string} [vKey='v'] + * @memberof heatmap# * @property */ - grouping, + vKey = 'v', + /** - * How to get the value of the violin - * @param {function} [valueExtractor=function(key, index) { return data[key] }] - * @memberof violin# + * Function for extracting the the value from xKey. + * (see {@link heatmap.xExtractor}) + * @param {function} [xExtractor=function(key, i) { return data[key][xKey] }] + * @returns {string} + * @memberof heatmap# * @property */ - valueExtractor = function(key, index) {return data[key] }, + xExtractor = function(key, i) {return data[key][xKey] }, /** - * How to sort the bars - if {@link violin#grouping} is not provided. - * @param {function} [sortingFunction=function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}] - * @memberof violin# + * Function for extracting the the value from yKey. + * (see {@link heatmap.yExtractor}) + * @param {function} [yExtractor=function(key, i) { return data[key][yKey] }] + * @returns {string} + * @memberof heatmap# * @property */ - sortingFunction = function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}, + yExtractor = function(key, i) { return data[key][yKey] }, /** - * The scale for which violin values should be transformed by - * @param {d3.scale} [scale=d3.scaleLinear] - * @memberof violin# + * Function for extracting the the value from vKey. + * (see {@link heatmap.vExtractor}) + * @param {function} [vExtractor=function(key, i) { return data[key][vKey] }] + * @returns {number} + * @memberof heatmap# * @property */ - scale = d3.scaleLinear(), + vExtractor = function(key, i) { return data[key][vKey] }, + + /** - * The padding for the domain of the scale (see {@link violin#scale}) - * @param {number} [domainPadding=0.5] - * @memberof violin# + * Whether or not to allow heatmap to render elements pass the main spatial dimension + * given the orientation (see {@link heatmap#orient}), where {@link heatmap#orient}="bottom" or {@link heatmap#orient}="top" + * the main dimension is {@link heatmap#spaceX} and where {@link heatmap#orient}="left" or {@link heatmap#orient}="right" + * the main dimension is {@link heatmap#spaceY} + * @param {boolean} [overflowQ=false] + * @memberof heatmap# * @property */ - domainPadding = 0.5, + overflowQ = false, + /** * Default space for the spacer (percentage) of main dimension given the orientation - * (see {@link violin#orient}), where {@link violin#orient}="horizontal" - * the main dimension is {@link violin#spaceX} and where {@link violin#orient}="vertical" - * the main dimension is {@link violin#spaceY} between bars - * @param {number} [objectSpacer=0.05] - * @memberof violin# + * (see {@link heatmap#orient}), where {@link heatmap#orient}="horizontal" + * the main dimension is {@link heatmap#spaceX} and where {@link heatmap#orient}="vertical" + * the main dimension is {@link heatmap#spaceY} between bubbles + * @param {number} [objectSpacer=0.0] + * @memberof heatmap# * @property */ - objectSpacer = 0.05, + objectSpacer = 0.0, /** - * The minimum size that an object can be + * The minimum size that an object can be in the y dimension * @param {number} [minObjectSize=50] - * @memberof violin# - * @property - */ - minObjectSize = 50, - /** - * The maximum size that an object can be - * @param {number} [maxObjectSize=100] - * @memberof violin# + * @memberof heatmap# * @property */ - maxObjectSize = 100, - + yMinObjectSize = 50, /** - * The stroke width of the bars - * @param {number} [barStrokeWidth=2] - * @memberof violin# + * The minimum size that an object can be in the x dimension + * @param {number} [minObjectSize=50] + * @memberof heatmap# * @property */ - objectStrokeWidth = 2, + xMinObjectSize = 50, /** - * Instance of ColorFunction - * @param {function} [colorFunction = colorFunction()] - * @memberof violin# + * The maximum size that an object can be in the x dimension + * @param {number} [maxObjectSize=100] + * @memberof heatmap# * @property */ - colorFunction$$1 = colorFunction(), + xMaxObjectSize = 100, /** - * Instance of ColorFunction modified by a scale for the points - * @param {function} [pointColorFunc = colorFunction()] - * @memberof violin# + * The maximum size that an object can be in the y dimension + * @param {number} [maxObjectSize=100] + * @memberof heatmap# * @property */ - pointColorFunc = function (d, type, base, min, max) { - var minMaxHexScale = d3.scaleLinear().domain([min, max]).range([-0.25, 0.05]); - var scaledColor = utils.color.modifyHexidecimalColorLuminance(base.replace('#', ''), minMaxHexScale(d)); - var mod = type == "stroke" ? 0 : 0.25; - return utils.color.modifyHexidecimalColorLuminance(scaledColor.replace('#', ''), mod) - }, + yMaxObjectSize = 100, + /** - * The radius of a point - * @param {number} [pointRadius=3] - * @memberof violin# - * @property - */ - pointRadius = 3, - /** - * The stroke width of the oints - * @param {number} [pointStrokeWidth=2] - * @memberof violin# + * The stroke width of the bubbles + * @param {number} [objectStrokeWidth=2] + * @memberof heatmap# * @property */ - pointStrokeWidth = 2, + objectStrokeWidth = 2, + // colorFunc = colorFunction(), /** * Color of the background * @param {string} [backgroundFill="transparent"] - * @memberof violin# + * @memberof heatmap# * @property */ backgroundFill = 'transparent', /** - * Namespace for all items made by this instance of violin - * @param {string} [namespace="d3sm-violin"] - * @memberof violin# + * Namespace for all items made by this instance of heatmap + * @param {string} [namespace="d3sm-heatmap"] + * @memberof heatmap# * @property */ - namespace = 'd3sm-violin', + namespace = 'd3sm-heatmap', /** - * Class name for violin container ( element) - * @param {string} [objectClass="violin"] - * @memberof violin# + * Class name for heatmap container ( element) + * @param {string} [objectClass="heatmap"] + * @memberof heatmap# * @property */ - objectClass = 'violin', + objectClass = 'heatmap', /** * Duration of all transitions of this element * @param {number} [transitionDuration=1000] - * @memberof violin# + * @memberof heatmap# * @property */ transitionDuration = 1000, /** * Easing function for transitions * @param {d3.ease} [easeFunc=d3.easeExp] - * @memberof violin# + * @memberof heatmap# * @property */ easeFunc = d3.easeExp, /** - * The keys corresponding to each quartile - * @param {string[]} [quartileKeys=["Q0", "Q1", "Q2", "Q3", "Q4"]] - * @memberof violin# + * Stores the keys of all the cells + * Calculated after heatmap called. + * @param {string[]} [cellKeys=undefined] + * @memberof heatmap# * @property */ - quartileKeys = ["Q0", "Q1", "Q2", "Q3", "Q4"], - + cellKeys, /** - * The keys of the bars - * @param {string[]} [violinKeys=undefined] - * @memberof violin# + * Stores the list of utils.arr.unique xValues + * Calculated after heatmap called. + * @param {string[]} [xValues=undefined] + * @memberof heatmap# * @property */ - violinKeys, + xValues, /** - * The values of the bars - * @param {number[]} [violinValues=undefined] - * @memberof violin# + * Stores the list of utils.arr.unique yValues + * Calculated after heatmap called. + * @param {string[]} [yValues=undefined] + * @memberof heatmap# * @property */ - violinValues, + yValues, /** - * The objectSize (actual width) used by the bars - * @param {number} [objectSize=undefined] - * @memberof violin# + * Stores the list of utils.arr.unique vValues + * Calculated after heatmap called. + * @param {string[]} [vValues=undefined] + * @memberof heatmap# * @property */ - objectSize, + vValues, + + xKeySortingFunction = function(a, b) { return xValues.indexOf(xExtractor(a)) - xValues.indexOf(xExtractor(b)) }, + yKeySortingFunction = function(a, b) { return yValues.indexOf(yExtractor(a)) - yValues.indexOf(yExtractor(b)) }, /** - * The spacerSize (actual width) used by the spacers between the bars - * @param {number} [spacerSize=undefined] - * @memberof violin# + * Instance of ColorFunction with .colorBy set to 'category' + * @function colorFunction + * @memberof heatmap# * @property */ - spacerSize, - + colorFunction$$1 = colorFunction().colorBy('category'), /** * Instance of Tooltip - * @param {function} [tooltip=tooltip()] - * @memberof violin# + * @function tooltip + * @memberof heatmap# * @property */ - tooltip$$1 = tooltip().keys([quartileKeys[4], quartileKeys[3], quartileKeys[2], quartileKeys[1], quartileKeys[0]]), - pointsTooltip = tooltip(), - // pointKeyExtractor = function(violinKey, violinData, violinValues) {return d3.keys(violinValues[violinKey].values)}, - // pointValueExtractor = function(pointKey, violinKey, violinData, violinValues) {return violinValues[violinKey].values[pointKey]}, - + tooltip$$1 = tooltip(), /** - * Function which given the key of the violin and that key's associated value - * returns the object consiting of the points of the violin - * (see {@link violin#violinPointsExtractor}) - * @param {Object} [violinPointsExtractor=function(violinKey, violinData) { return violinData.points }] - * @memberof violin# + * store the size the heatmap could be in the x dimension + * the actuall size of the heatmap will be the min of xSize and {@link heatmap#ySize} + * Calculated after heatmap called. + * @param {string[]} [xSize=undefined] + * @memberof heatmap# * @property */ - violinPointsExtractor = function (violinKey, violinData) {return violinData.points }, + xSize, /** - * Function which given the key of the current point and the object of points for the - * violin, returns the numerical value of the point - * (see {@link violin#violinPointValueExtractor}) - * @param {Object} [violinPointValueExtractor=function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value }] - * @memberof violin# + * store the size of the spacer in the x dimension + * Calculated after heatmap called. + * @param {string[]} [xSpacerSize=undefined] + * @memberof heatmap# * @property */ - violinPointValueExtractor = function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value }; - + xSpacerSize, /** - * Gets or sets the violinPointsExtractor - * @param {function} [_=none] - * @returns {violin | function} - * @memberof violin - * @property - * by default violinPointsExtractor = function(violinKey, violinData) { return violinData.points } - */ - violin.violinPointsExtractor = function(_) { return arguments.length ? (violinPointsExtractor = _, violin) : violinPointsExtractor; }; + * store the size the heatmap could be in the y dimension + * the actuall size of the heatmap will be the min of xSize and {@link heatmap#xSize} + * Calculated after heatmap called. + * @param {string[]} [ySize=undefined] + * @memberof heatmap# + * @property + */ + ySize, /** - * Gets or sets the violinPointValueExtractor - * @param {function} [_=none] - * @returns {violin | function} - * @memberof violin - * @property - * by default violinPointsExtractor = function(pointKey, violinKey, violinData, violinValues) {return violinValues[violinKey].values[pointKey]} - */ - violin.violinPointValueExtractor = function(_) { return arguments.length ? (violinPointValueExtractor = _, violin) : violinPointValueExtractor; }; - + * store the size of the spacer in the y dimension. + * Calculated after heatmap called. + * @param {string[]} [xSpacerSize=undefined] + * @memberof heatmap# + * @property + */ + ySpacerSize; /** * Gets or sets the selection in which items are manipulated * @param {d3.selection} [_=none] - * @returns {violin | d3.selection} - * @memberof violin + * @returns {heatmap | d3.selection} + * @memberof heatmap * @property * by default selection = selection */ - violin.selection = function(_) { return arguments.length ? (selection = _, violin) : selection; }; + hm.selection = function(_) { return arguments.length ? (selection = _, hm) : selection; }; /** * Gets or sets the data - * (see {@link violin#data}) - * @param {number} [_=none] - * @returns {violin | object} - * @memberof violin - * @property - */ - violin.data = function(_) { return arguments.length ? (data = _, violin) : data; }; - /** - * Gets or sets the orient of the boxes - * (see {@link violin#orient}) + * (see {@link heatmap#data}) * @param {number} [_=none] - * @returns {violin | object} - * @memberof violin + * @returns {heatmap | object} + * @memberof heatmap * @property */ - violin.orient = function(_) { return arguments.length ? (orient = _, violin) : orient; }; + hm.data = function(_) { return arguments.length ? (data = _, hm) : data; }; + // hm.orient = function(_) { return arguments.length ? (orient = _, hm) : orient; } /** * Gets or sets the amount of horizontal space in which items are manipulated - * (see {@link violin#spaceX}) + * (see {@link heatmap#spaceX}) * @param {number} [_=none] should be a number > 0 - * @returns {violin | number} - * @memberof violin + * @returns {heatmap | number} + * @memberof heatmap * @property * by default spaceX = undefined */ - violin.spaceX = function(_) { return arguments.length ? (spaceX = _, violin) : spaceX; }; + hm.spaceX = function(_) { return arguments.length ? (spaceX = _, hm) : spaceX; }; /** * Gets or sets the amount of vertical space in which items are manipulated - * (see {@link violin#spaceY}) + * (see {@link heatmap#spaceY}) * @param {number} [_=none] should be a number > 0 - * @returns {violin | number} - * @memberof violin + * @returns {heatmap | number} + * @memberof heatmap * @property * by default spaceY = undefined */ - violin.spaceY = function(_) { return arguments.length ? (spaceY = _, violin) : spaceY; }; - + hm.spaceY = function(_) { return arguments.length ? (spaceY = _, hm) : spaceY; }; /** - * Gets / sets whether or not violin is allowed to go beyond specified dimensions - * (see {@link violin#overflowQ}) - * @param {boolean} [_=none] - * @returns {violin | boolean} - * @memberof violin + * Gets or sets the xKey + * (see {@link heatmap#xKey}) + * @param {string} [_=none] + * @returns {heatmap | string} + * @memberof heatmap * @property - * by default overflowQ = false + * by default xKey = 'x' */ - violin.overflowQ = function(_) { return arguments.length ? (overflowQ = _, violin) : overflowQ; }; + hm.xKey = function(_) { return arguments.length ? (xKey = _, hm) : xKey; }; /** - * Gets / sets whether or not to plot points with the violins - * (see {@link violin#pointsQ}) - * @param {boolean} [_=none] - * @returns {violin | boolean} - * @memberof violin + * Gets or sets the yKey + * (see {@link heatmap#yKey}) + * @param {string} [_=none] + * @returns {heatmap | string} + * @memberof heatmap * @property - * by default pointsQ = false + * by default yKey = 'y' */ - violin.pointsQ = function(_) { return arguments.length ? (pointsQ = _, violin) : pointsQ; }; - + hm.yKey = function(_) { return arguments.length ? (yKey = _, hm) : yKey; }; /** - * Gets / sets the grouping of the boxes - * (see {@link violin#grouping}) - * @param {Array[]} [_=none] - * @returns {violin | Array[]} - * @memberof violin + * Gets or sets the vKey + * (see {@link heatmap#vKey}) + * @param {string} [_=none] + * @returns {heatmap | string} + * @memberof heatmap * @property - * by default grouping = undefined + * by default vKey = 'y' */ - violin.grouping = function(_) { return arguments.length ? (grouping = _, violin) : grouping; }; + hm.vKey = function(_) { return arguments.length ? (vKey = _, hm) : vKey; }; + /** - * Gets / sets the valueExtractor - * (see {@link violin#valueExtractor}) - * @param {function} [_=none] - * @returns {violin | function} - * @memberof violin + * Gets or sets the cellKeys + * (see {@link heatmap#cellKeys}) + * @param {string[]} [_=none] + * @returns {heatmap | string[]} + * @memberof heatmap * @property + * by default cellKeys = undefined */ - violin.valueExtractor = function(_) { return arguments.length ? (valueExtractor = _, violin) : valueExtractor; }; + hm.cellKeys = function(_) { return arguments.length ? (cellKeys = _, hm) : cellKeys; }; /** - * Gets / sets the sortingFunction - * (see {@link violin#sortingFunction}) - * @param {function} [_=none] - * @returns {violin | function} - * @memberof violin + * Gets or sets the xValues + * (see {@link heatmap#xValues}) + * @param {string[]} [_=none] + * @returns {heatmap | string[]} + * @memberof heatmap * @property + * by default xValues = undefined */ - violin.sortingFunction = function(_) { return arguments.length ? (sortingFunction = _, violin) : sortingFunction; }; - + hm.xValues = function(_) { return arguments.length ? (xValues = _, hm) : xValues; }; /** - * Gets / sets the scale for which the violin values should be transformed by - * (see {@link violin#scale}) - * @param {d3.scale} [_=none] - * @returns {violin | d3.scale} - * @memberof violin + * Gets or sets the yValues + * (see {@link heatmap#yValues}) + * @param {string[]} [_=none] + * @returns {heatmap | string[]} + * @memberof heatmap * @property - * by default scale = d3.scaleLinear() + * by default yValues = undefined */ - violin.scale = function(_) { return arguments.length ? (scale = _, violin) : scale; }; + hm.yValues = function(_) { return arguments.length ? (yValues = _, hm) : yValues; }; /** - * Gets / sets the padding for the domain of the scale - * (see {@link violin#domainPadding}) - * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin + * Gets or sets the vValues + * (see {@link heatmap#vValues}) + * @param {number[]} [_=none] + * @returns {heatmap | number[]} + * @memberof heatmap * @property - * by default domainPadding = 0.5 + * by default vValues = undefined */ - violin.domainPadding = function(_) { return arguments.length ? (domainPadding = _, violin) : domainPadding; }; + hm.vValues = function(_) { return arguments.length ? (vValues = _, hm) : vValues; }; /** - * Gets / sets objectSpacer - * (see {@link violin#objectSpacer}) - * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin + * Gets or sets the xExtractor + * (see {@link heatmap#xExtractor}) + * @param {function} [_=none] + * @returns {heatmap | function} + * @memberof heatmap * @property - * by default objectSpacer = 0.05 + * by default xExtractor = undefined */ - violin.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, violin) : objectSpacer; }; + hm.xExtractor = function(_) { return arguments.length ? (xExtractor = _, hm) : xExtractor; }; /** - * Gets / sets the minObjectSize - * (see {@link violin#minObjectSize}) - * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin + * Gets or sets the yExtractor + * (see {@link heatmap#yExtractor}) + * @param {function} [_=none] + * @returns {heatmap | function} + * @memberof heatmap * @property - * by default minObjectSize = 15 + * by default yExtractor = undefined */ - violin.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, violin) : minObjectSize; }; + hm.yExtractor = function(_) { return arguments.length ? (yExtractor = _, hm) : yExtractor; }; /** - * Gets / sets the maxObjectSize - * (see {@link violin#maxObjectSize}) - * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin + * Gets or sets the vExtractor + * (see {@link heatmap#vExtractor}) + * @param {function} [_=none] + * @returns {heatmap | function} + * @memberof heatmap * @property - * by default maxObjectSize = 50 + * by default vExtractor = undefined */ - violin.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, violin) : maxObjectSize; }; + hm.vExtractor = function(_) { return arguments.length ? (vExtractor = _, hm) : vExtractor; }; /** - * Gets / sets the objectStrokeWidth - * (see {@link violin#objectStrokeWidth}) + * Gets / sets whether or not heatmap is allowed to go beyond specified dimensions + * (see {@link heatmap#spaceX}) + * @param {boolean} [_=none] + * @returns {heatmap | boolean} + * @memberof heatmap + * @property + * by default overflowQ = false + */ + hm.overflowQ = function(_) { return arguments.length ? (overflowQ = _, hm) : overflowQ; }; + /** + * Gets / sets objectSpacer + * (see {@link heatmap#objectSpacer}) * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin + * @returns {heatmap | number} + * @memberof heatmap * @property - * by default objectStrokeWidth = 2 + * by default objectSpacer = 0.0 */ - violin.objectStrokeWidth = function(_) { return arguments.length ? (objectStrokeWidth = _, violin) : objectStrokeWidth; }; - - + hm.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, objectSpacer) : data; }; /** - * Gets / sets the colorFunction - * (see {@link violin#colorFunction}) - * @param {colorFunction} [_=none] - * @returns {violin | colorFunction} - * @memberof violin + * Gets / sets the minObjectSize + * (see {@link heatmap#minObjectSize}) + * @param {number} [_=none] + * @returns {heatmap | number} + * @memberof heatmap * @property - * by default colorFunction = colorFunction() + * by default minObjectSize = 50 */ - violin.colorFunction = function(_) { return arguments.length ? (colorFunction$$1 = _, violin) : colorFunction$$1; }; + hm.yMinObjectSize = function(_) { return arguments.length ? (yMinObjectSize = _, hm) : yMinObjectSize; }; /** - * Gets / sets the colorFunction - * (see {@link violin#colorFunction}) - * @param {colorFunction} [_=none] - * @returns {violin | colorFunction} - * @memberof violin + * Gets / sets the maxObjectSize + * (see {@link heatmap#maxObjectSize}) + * @param {number} [_=none] + * @returns {heatmap | number} + * @memberof heatmap * @property - * by default colorFunction = colorFunction() + * by default maxObjectSize = 100 */ - violin.pointColorFunc = function(_) { return arguments.length ? (pointColorFunc = _, violin) : pointColorFunc; }; - - + hm.yMaxObjectSize = function(_) { return arguments.length ? (yMaxObjectSize = _, hm) : yMaxObjectSize; }; /** - * Gets / sets the pointRadius - * (see {@link violin#pointRadius}) + * Gets / sets the minObjectSize + * (see {@link heatmap#minObjectSize}) * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin + * @returns {heatmap | number} + * @memberof heatmap * @property - * by default pointRadius = 2 + * by default minObjectSize = 50 */ - violin.pointRadius = function(_) { return arguments.length ? (pointRadius = _, violin) : pointRadius; }; + hm.xMinObjectSize = function(_) { return arguments.length ? (xMinObjectSize = _, hm) : xMinObjectSize; }; /** - * Gets / sets the pointStrokeWidth - * (see {@link violin#pointStrokeWidth}) + * Gets / sets the maxObjectSize + * (see {@link heatmap#maxObjectSize}) * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin + * @returns {heatmap | number} + * @memberof heatmap * @property - * by default pointStrokeWidth = 2 + * by default maxObjectSize = 100 */ - violin.pointStrokeWidth = function(_) { return arguments.length ? (pointStrokeWidth = _, violin) : pointStrokeWidth; }; - - + hm.xMaxObjectSize = function(_) { return arguments.length ? (xMaxObjectSize = _, hm) : xMaxObjectSize; }; + /** + * Gets / sets the objectStrokeWidth + * (see {@link heatmap#objectStrokeWidth}) + * @param {number} [_=none] + * @returns {heatmap | number} + * @memberof heatmap + * @property + * by default objectStrokeWidth = 2 + */ + hm.objectStrokeWidth = function(_) { return arguments.length ? (objectStrokeWidth = _, hm) : objectStrokeWidth; }; /** * Gets / sets the backgroundFill - * (see {@link violin#backgroundFill}) + * (see {@link heatmap#backgroundFill}) * @param {string} [_=none] - * @returns {violin | string} - * @memberof violin + * @returns {heatmap | string} + * @memberof heatmap * @property * by default backgroundFill = 'transparent' */ - violin.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, violin) : backgroundFill; }; + hm.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, hm) : backgroundFill; }; /** * Gets / sets the namespace - * (see {@link violin#namespace}) + * (see {@link heatmap#namespace}) * @param {string} [_=none] - * @returns {violin | string} - * @memberof violin + * @returns {heatmap | string} + * @memberof heatmap * @property - * by default namespace = 'd3sm-violin' + * by default namespace = 'd3sm-heatmap' */ - violin.namespace = function(_) { return arguments.length ? (namespace = _, violin) : namespace; }; + hm.namespace = function(_) { return arguments.length ? (namespace = _, hm) : namespace; }; /** * Gets / sets the objectClass - * (see {@link violin#objectClass}) + * (see {@link heatmap#objectClass}) * @param {string} [_=none] - * @returns {violin | string} - * @memberof violin + * @returns {heatmap | string} + * @memberof heatmap * @property * by default objectClass = 'tick-group' */ - violin.objectClass = function(_) { return arguments.length ? (objectClass = _, violin) : objectClass; }; - - + hm.objectClass = function(_) { return arguments.length ? (objectClass = _, hm) : objectClass; }; /** * Gets / sets the transitionDuration - * (see {@link violin#transitionDuration}) + * (see {@link heatmap#transitionDuration}) * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin + * @returns {heatmap | number} + * @memberof heatmap * @property * by default transitionDuration = 1000 */ - violin.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, violin) : transitionDuration; }; + hm.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, hm) : transitionDuration; }; /** * Gets / sets the easeFunc - * (see {@link violin#easeFunc}) + * (see {@link heatmap#easeFunc}) * @param {d3.ease} [_=none] - * @returns {violin | d3.ease} - * @memberof violin + * @returns {heatmap | d3.ease} + * @memberof heatmap * @property * by default easeFunc = d3.easeExp */ - violin.easeFunc = function(_) { return arguments.length ? (easeFunc = _, violin) : easeFunc; }; - + hm.easeFunc = function(_) { return arguments.length ? (easeFunc = _, hm) : easeFunc; }; /** - * Gets / sets the quartileKey - * (see {@link violin#quartileKey}) - * @param {string} [_=none] - * @returns {violin | string} - * @memberof violin + * Gets / sets the tooltip + * (see {@link heatmap#tooltip}) + * @param {tooltip} [_=none] + * @returns {heatmap | tooltip} + * @memberof heatmap * @property - * by default quartileKey = "utils.math.quartiles" + * by default tooltip = tooltip() */ - violin.quartileKey = function(_) { return arguments.length ? (quartileKey = _, violin) : quartileKey; }; + hm.tooltip = function(_) { return arguments.length ? (tooltip$$1 = _, hm) : tooltip$$1; }; + /** - * Gets / sets the quartileKeys - * (see {@link violin#quartileKeys}) - * @param {string[]} [_=none] - * @returns {violin | string[]} - * @memberof violin + * Gets / sets the colorFunction + * (see {@link heatmap#colorFunction}) + * @param {colorFunction} [_=none] + * @returns {heatmap | colorFunction} + * @memberof heatmap * @property - * by default quartileKeys = ["Q0","Q1","Q2","Q3","Q4"] + * by default colorFunction = colorFunction() */ - violin.quartileKeys = function(_) { return arguments.length ? (quartileKeys = _, violin) : quartileKeys; }; - + hm.colorFunction = function(_) { return arguments.length ? (colorFunction$$1 = _, hm) : colorFunction$$1; }; /** - * Gets / sets the violinKeys - * (see {@link violin#violinKeys}) - * @param {string[]} [_=none] - * @returns {violin | string[]} - * @memberof violin + * Gets / sets the xSize + * (see {@link heatmap#xSize}) + * @param {number} [_=none] + * @returns {heatmap | number} + * @memberof heatmap * @property - * by default violinKeys = undefined + * by default xSize = undefined */ - violin.violinKeys = function(_) { return arguments.length ? (violinKeys = _, violin) : violinKeys; }; + hm.xSize = function(_) { return arguments.length ? (xSize = _, hm) : xSize; }; /** - * Gets / sets the violinValues - * (see {@link violin#violinValues}) - * @param {Object[]} [_=none] - * @returns {violin | Object[]} - * @memberof violin + * Gets / sets the xSpacerSize + * (see {@link heatmap#xSpacerSize}) + * @param {number} [_=none] + * @returns {heatmap | number} + * @memberof heatmap * @property - * by default violinValues = undefined + * by default xSpacerSize = undefined */ - violin.violinValues = function(_) { return arguments.length ? (violinValues = _, violin) : violinValues; }; - + hm.xSpacerSize = function(_) { return arguments.length ? (xSpacerSize = _, hm) : xSpacerSize; }; /** - * Gets / sets the objectSize - * (see {@link violin#objectSize}) + * Gets / sets the ySize + * (see {@link heatmap#ySize}) * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin + * @returns {heatmap | number} + * @memberof heatmap * @property - * by default objectSize = undefined + * by default ySize = undefined */ - violin.objectSize = function(_) { return arguments.length ? (objectSize = _, violin) : objectSize; }; + hm.ySize = function(_) { return arguments.length ? (ySize = _, hm) : ySize; }; /** - * Gets / sets the spacerSize - * (see {@link violin#spacerSize}) + * Gets / sets the ySpacerSize + * (see {@link heatmap#ySpacerSize}) * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin + * @returns {heatmap | number} + * @memberof heatmap * @property - * by default spacerSize = undefined + * by default ySpacerSize = undefined */ - violin.spacerSize = function(_) { return arguments.length ? (spacerSize = _, violin) : spacerSize; }; - /** - * Gets / sets the tooltip - * (see {@link violin#tooltip}) - * @param {tooltip} [_=none] - * @returns {violin | tooltip} - * @memberof violin - * @property - * by default tooltip = tooltip() - */ - violin.tooltip = function(_) { return arguments.length ? (tooltip$$1 = _, violin) : tooltip$$1; }; - /** - * Gets / sets the pointsTooltip - * (see {@link violin#pointsTooltip}) - * @param {tooltip} [_=none] - * @returns {violin | tooltip} - * @memberof violin - * @property - * by default pointsTooltip = tooltip() - */ - violin.pointsTooltip = function(_) { return arguments.length ? (pointsTooltip = _, violin) : pointsTooltip; }; - - - + hm.ySpacerSize = function(_) { return arguments.length ? (ySpacerSize = _, hm) : ySpacerSize; }; + // hm.yKeySortingFunction = function(_) { return arguments.length ? (yKeySortingFunction = _, hm) : yKeySortingFunction; } + // hm.xKeySortingFunction = function(_) { return arguments.length ? (xKeySortingFunction = _, hm) : xKeySortingFunction; } - function violin () { - // for convenience in handling orientation specific values - var horizontalQ = (orient == 'horizontal') ? true : false; - // background cliping rectangle + function hm() { var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); - // if grouping is undefined sort violinKeys by sortingFunction - var ordered = (grouping == undefined) ? d3.keys(data).sort(sortingFunction) : grouping; - - // console.log(ordered) - - violinKeys = utils.arr.flatten(ordered); - + cellKeys = d3.keys(data); - var calcValues = neededViolinValues() - .horizontalQ(horizontalQ) - .quartileKeys(quartileKeys) - .violinPointsExtractor(violinPointsExtractor) - .violinPointValueExtractor(violinPointValueExtractor); + xValues = utils.arr.unique(cellKeys.map(xExtractor)); + yValues = utils.arr.unique(cellKeys.map(yExtractor)); + vValues = utils.arr.unique(cellKeys.map(vExtractor)); + cellKeys.sort(function(a, b){ return xKeySortingFunction(a, b) || yKeySortingFunction(a, b) }); + utils.con.log('heatmap', 'cells are sorted by', cellKeys); - // augment valus - violinKeys.map(function(vk, i){ calcValues(vk, data); }); - var numberOfObjects = violinKeys.length; + utils.con.log('heatmap', 'x and y keys are', {x: xValues, y:yValues}); - var min = [].concat(...violinKeys.map(function(k, i){return data[k].utils.math.quartiles[quartileKeys[0]]})); - var max = [].concat(...violinKeys.map(function(k, i){return data[k].utils.math.quartiles[quartileKeys[quartileKeys.length - 1]]})); - var extent = [Math.min(...min) - domainPadding, Math.max(...max) + domainPadding]; - // console.log(extent, violinValues, ordered) - // set the scale - scale.domain(extent).range(horizontalQ ? [0,spaceY] : [0, spaceX]); - var space = horizontalQ ? spaceX : spaceY; - // calculate object size - objectSize = utils.math.calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ); - // calculate spacer size if needed - spacerSize = utils.math.calculateWidthOfSpacer(ordered, space, objectSize, numberOfObjects, objectSpacer, overflowQ); - // make the nested groups - var spacerFunction = groupingSpacer() - .horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects) - .objectClass(objectClass).objectSize(objectSize).spacerSize(spacerSize) - .transitionDuration(transitionDuration).easeFunc(easeFunc) - .namespace(namespace); + var xDim = xValues.length, yDim = yValues.length; - // move stuff - spacerFunction(container, ordered, 0); - // console.log(violinKeys, ordered, container.selectAll('g:not(.to-remove).'+objectClass).nodes()) - // for color function - var parentIndexArray = []; - container.selectAll('g:not(.to-remove).'+objectClass) - .each(function(d, i){if (utils.arr.hasQ(violinKeys, d)){ parentIndexArray.push(Number(d3.select(this).attr('parent-index')));}}); + ySize = utils.math.calculateWidthOfObject(spaceY, yDim, yMinObjectSize, yMaxObjectSize, objectSpacer, overflowQ); + xSize = utils.math.calculateWidthOfObject(spaceX, xDim, xMinObjectSize, xMaxObjectSize, objectSpacer, overflowQ); + ySpacerSize = utils.math.calculateWidthOfSpacer(yValues, spaceY, ySize, yDim, objectSpacer, overflowQ); + xSpacerSize = utils.math.calculateWidthOfSpacer(xValues, spaceX, xSize, xDim, objectSpacer, overflowQ); + // console.table({ + // x:{ + // object: xSize, + // spacer: xSpacerSize, + // dim: xDim + // }, + // y:{ + // object: ySize, + // spacer: ySpacerSize, + // dim: yDim + // } + // + // }) + utils.con.log('heatmap', 'size of', {x: xSize, y: ySize}); - // update color function - colorFunction$$1 = colorFunction$$1.colorBy() == 'index' - ? colorFunction$$1.dataExtent([0, Math.max(...parentIndexArray)]) - : colorFunction$$1.dataExtent(extent); - /* violiin specific needs */ + var ySpacer = groupingSpacer() + .horizontalQ(false) + .moveby('category') + .numberOfObjects(yDim) + .objectClass(utils.str.hypenate(objectClass, 'row')) + .objectSize(ySize + ySpacerSize) + .spacerSize(0) + .transitionDuration(transitionDuration) + .easeFunc(easeFunc) + .namespace('row'); - var frequencyMax = Math.max(...[].concat(...violinKeys.map(function(k, i){return d3.max(data[k].frequencies)}))); - var vScale = d3.scaleLinear().domain([0, frequencyMax]).range([0, objectSize / 2]); + var xSpacer = groupingSpacer() + .horizontalQ(true) + .moveby('category') + .numberOfObjects(xDim) + .objectClass(objectClass) + .objectSize(xSize + xSpacerSize) + .spacerSize(0) + .transitionDuration(transitionDuration) + .easeFunc(easeFunc); - var lArea = d3.line() - .x(function(d, i){ return horizontalQ ? -vScale(d.x) : scale(d.x)}) - .y(function(d, i){ return horizontalQ ? scale(extent[1]) - scale(d.y) : -vScale(d.y)}) - .curve(d3.curveBasis); - var rArea = d3.line() - .x(function(d, i){ return horizontalQ ? vScale(d.x) : scale(d.x)}) - .y(function(d, i){ return horizontalQ ? scale(extent[1]) - scale(d.y) : vScale(d.y)}) - .curve(d3.curveBasis); + ySpacer(container, yValues, 0); + container.selectAll('g.'+utils.str.hypenate(objectClass, 'row')) + .each(function(d, i){ xSpacer(d3.select(this), xValues, 0); }); + var cells = container.selectAll('g:not(.to-remove).'+objectClass); + if (cellKeys.length != yValues.length * xValues.length) { + var lookup = {}; + cellKeys.map(function(k, i){ + lookup[xExtractor(k)+'::'+yExtractor(k)] = k; + }); + var positionedCellKeys = []; + for (var i = 0; i < yValues.length; i++) { + for (var j = 0; j < xValues.length; j++) { + var lookupValue = lookup[xValues[j]+"::"+yValues[i]]; + if (lookupValue == undefined) { + positionedCellKeys.push(undefined); + } else { + positionedCellKeys.push(lookupValue); + } + } + } - container.selectAll('g:not(.to-remove).'+objectClass).each(function(key, i){ - var t = d3.select(this), - currentData = data[key]; - // needed because bug in selecting .to-remove - if (!utils.arr.hasQ(violinKeys, key)) {return} - var - i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'), - fillColor = colorFunction$$1(key, currentData, i, 'fill'), // prevent duplicate computation - strokeColor = colorFunction$$1(key, currentData, i, 'stroke'), - area = utils.sel.safeSelect(t, 'g', 'area'), - la = utils.sel.safeSelect(area, 'path', 'left'), - ra = utils.sel.safeSelect(area, 'path', 'right'), - quarts = utils.sel.safeSelect(t, 'g', 'quarts'), - lq3 = utils.sel.safeSelect(quarts, 'line', 'q3'), - lq1 = utils.sel.safeSelect(quarts, 'line', 'q1'), - q3 = currentData.utils.math.quartiles[quartileKeys[3]], - q2 = currentData.utils.math.quartiles[quartileKeys[2]], - q1 = currentData.utils.math.quartiles[quartileKeys[1]]; + cells.data(positionedCellKeys); - t.attr('transform', horizontalQ ? 'translate('+objectSize / 2+',0)' : 'translate(0,'+objectSize / 2+')' ); - // draw curve - la.transition().duration(transitionDuration).attr('d', function(dd, ii){ return lArea(currentData.contour)}) - .attr('fill', fillColor) - .attr('stroke', strokeColor) - .attr('stroke-width', objectStrokeWidth); + // maybe breaks this + // !!!!!! IMPORTANT NOTE TODO LOOK HERE BUG + cellKeys = positionedCellKeys; + } else { + cells.data(cellKeys); + } - ra.transition().duration(transitionDuration).attr('d', function(dd, ii){ return rArea(currentData.contour)}) - .attr('fill', fillColor) - .attr('stroke', strokeColor) - .attr('stroke-width', objectStrokeWidth); - area.node().addEventListener('mouseover', function(dd, ii){ - container.selectAll('g.'+objectClass).style('opacity', 0.2); - t.style('opacity', 1); - la.attr('stroke-width',objectStrokeWidth*2); - ra.attr('stroke-width',objectStrokeWidth*2); - }); - area.node().addEventListener('mouseout', function(dd, ii){ - container.selectAll('g.'+objectClass).style('opacity', 1); - la.attr('stroke-width',objectStrokeWidth); - ra.attr('stroke-width',objectStrokeWidth); - }); - if (pointsQ) { - var ptsContainer = utils.sel.safeSelect(t, 'g', 'points'); - var pts = ptsContainer.selectAll('.point').data(currentData.pointKeys); - pts.on('mouseover', null); + var parentIndexArray = []; + cells.each(function(d, i){ parentIndexArray.push(Number(d3.select(this).attr('parent-index'))); }); + colorFunction$$1 = colorFunction$$1.colorBy() == 'index' + ? colorFunction$$1.dataExtent([0, Math.max(...parentIndexArray)]) + : colorFunction$$1.dataExtent([0, Math.max(...vValues)]); - var ptsExit = pts.exit().transition().ease(easeFunc).duration(transitionDuration) - .attr('r', 0) - .attr('cy', horizontalQ ? scale(extent[1]) - scale(q2) : vScale(0)) - .attr('cx', horizontalQ ? vScale(0) : scale(q2)).remove(); + cells.each(function(key, i) { + utils.con.log('heatmap', 'each cell', {key: key, index: i, node: d3.select(this).node()}); - var ptsEnter = pts.enter().append('circle').attr('class', 'point').attr('r', 0) - .attr('cx', horizontalQ ? 0 : scale(q2)) - .attr('cy', horizontalQ ? scale(q2) : 0); + var t = d3.select(this); + if (key == undefined) {return} + var currentData = data[key], + value = vExtractor(key, i), + i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'), + fillColor = colorFunction$$1(key, value, i, 'fill'), // prevent duplicate computation + strokeColor = colorFunction$$1(key, value, i, 'stroke'); - pts = pts.merge(ptsEnter); + var c = utils.sel.safeSelect(t, 'rect', utils.str.hypenate(objectClass,'rect')); + c.attr('width', xSize + xSpacerSize - objectStrokeWidth) + .attr('height', ySize + ySpacerSize - objectStrokeWidth) + .attr('fill', fillColor) + .attr('x', objectStrokeWidth/2) + .attr('y', objectStrokeWidth/2) + .attr('stroke', "#000") + .attr('stroke-width', objectStrokeWidth); - // console.log(pointsTooltip.header()) + }); - var pTTips = tooltip().selection(pts).data(violinPointsExtractor(key, currentData)) - .header(pointsTooltip.header()) - .keys(pointsTooltip.keys()) - .values(pointsTooltip.values()); + tooltip$$1.selection(cells.selectAll('rect.'+utils.str.hypenate(objectClass, 'rect'))) + .data(data); + // .keys(['r', 'v']) + // .header(function(d, i){return utils.str.hypenate(data[d][xKey], data[d][yKey]) }) - pTTips(); - - var pMin = d3.min(currentData.pointValues), pMax = d3.max(currentData.pointValues); - - - - pts.transition().duration(transitionDuration).ease(easeFunc).attr('r', pointRadius) - .attr('cy', function(pointKey, ii){ - var dd = currentData.pointValues[ii]; - if (horizontalQ) { return scale(extent[1]) - scale(dd) } - var j = utils.arr.whichBin(currentData.binned, dd); - var r = Math.random(); - var n = vScale(r * currentData.frequencies[j] * 0.5); - var k = Math.random() > 0.5 ? n : -n; - return k - }) - .attr('cx', function(pointKey, ii){ - var dd = currentData.pointValues[ii]; - if (horizontalQ) { - var j = utils.arr.whichBin(currentData.binned, dd); - var r = Math.random(); - var n = vScale(r * currentData.frequencies[j] * 0.5); - var k = Math.random() > 0.5 ? n : -n; - return k - } - return scale(dd) - }) - .attr('stroke', function(dd, ii) { var dd = currentData.pointValues[ii]; return pointColorFunc(dd, 'stroke', strokeColor, pMin, pMax) }) - .attr('fill' , function(dd, ii) { var dd = currentData.pointValues[ii]; return pointColorFunc(dd, 'fill' , strokeColor, pMin, pMax) }) - .attr('stroke-width', pointStrokeWidth); - - ptsContainer.selectAll('circle.point').on('mouseover', function(dd, ii){ - container.selectAll('g.'+objectClass).style('opacity', 0.2); - t.style('opacity', 1); - la.attr('stroke-width',objectStrokeWidth*2); - ra.attr('stroke-width',objectStrokeWidth*2); - - container.selectAll('.point').style('opacity', 0.2); - d3.select(this).style('opacity', 1).attr('r', pointRadius * 2).attr('stroke-width',pointStrokeWidth*2); - }); - ptsContainer.selectAll('circle.point').on('mouseout', function(dd, ii){ - var e = document.createEvent('SVGEvents'); - e.initEvent('mouseout',true,true); - area.node().dispatchEvent(e); - - container.selectAll('.point').style('opacity', 1); - d3.select(this).attr('stroke-width', pointStrokeWidth).attr('r', pointRadius); - }); - } - else { - cV.selectAll('.point') - .transition().duration(transitionDuration).ease(easeFunc) - .attr('r', 0) - .attr('cy', horizontalQ ? scale(extent[1]) - scale(q2) : vScale(0)) - .attr('cx', horizontalQ ? vScale(0) : scale(q2)) - .remove(); - } - - - }); - - - tooltip$$1.selection(container.selectAll('g:not(.to-remove).'+objectClass + ' .area')); - if (tooltip$$1.data() == undefined) {tooltip$$1.data(data);} - tooltip$$1(); - if (tooltip$$1.values() == undefined) { - tooltip$$1.values([ - function(currentData, tooltipKey){ return currentData['utils.math.quartiles'][tooltipKey] }, - function(currentData, tooltipKey){ return currentData['utils.math.quartiles'][tooltipKey] }, - function(currentData, tooltipKey){ return currentData['utils.math.quartiles'][tooltipKey] }, - function(currentData, tooltipKey){ return currentData['utils.math.quartiles'][tooltipKey] }, - function(currentData, tooltipKey){ return currentData['utils.math.quartiles'][tooltipKey] } - ]); - - } - - } - - return violin -} + tooltip$$1(); + } + return hm; +} +/******************************************************************************* +** ** +** ** +** VIOLIN ** +** ** +** ** +*******************************************************************************/ /** -* Produces the function which manipulates the violin data to have values needed -* for rendering the violins as svg. -* @returns {function} calculateViolinValues -* @namespace neededViolinValues -*/ -function neededViolinValues() { - var + * Creates a violin + * + * {@link https://sumneuron.gitlab.io/d3sm/demos/basic-violins/index.html Demo} + * @constructor violin + * @param {d3.selection} selection + * @namespace violin + * @returns {function} violin + */ +function violin( selection ) { + var /** - * Whether or not the orientation of the violins are horizontal + * Data to plot. Assumed to be a object, where each key corresponds to a violin + * (see {@link violin#data}) + * @param {Object} [data=undefined] + * @memberof violin# + * @property + */ + data, + /** + * Which direction to render the bars in * (see {@link violin#orient}) - * @param {Object} [horizontalQ=true] - * @memberof neededViolinValues# + * @param {number} [orient='horizontal'] + * @memberof violin# * @property */ - horizontalQ = true, + orient='horizontal', /** - * Keys to be put into the utils.math.quartiles if they need to be calculated. - * (see {@link violin#quartileKeys}) - * @param {Object} [quartileKeys=['Q0', 'Q1', 'Q2', 'Q3', 'Q4']] - * @memberof neededViolinValues# + * Amount of horizontal space (in pixels) avaible to render the violin in + * (see {@link violin#spaceX}) + * @param {number} [spaceX=undefined] + * @memberof violin# * @property */ - quartileKeys = ['Q0', 'Q1', 'Q2', 'Q3', 'Q4'], + spaceX, /** - * Function which given the key of the violin and that key's associated value - * returns the object consiting of the points of the violin - * (see {@link violin#violinPointsExtractor}) - * @param {Object} [violinPointsExtractor=function(violinKey, violinData) { return violinData.points }] - * @memberof neededViolinValues# + * Amount of vertical space (in pixels) avaible to render the violin in + * (see {@link violin.spaceY}) + * @param {number} [spaceY=undefined] + * @memberof violin# * @property */ - violinPointsExtractor, + spaceY, /** - * Function which given the key of the current point and the object of points for the - * violin, returns the numerical value of the point - * (see {@link violin#violinPointValueExtractor}) - * @param {Object} [violinPointValueExtractor=function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value }] - * @memberof neededViolinValues# + * Whether or not to allow violin to render elements pass the main spatial dimension + * given the orientation (see {@link violin#orient}), where {@link violin#orient}="horizontal" + * the main dimension is {@link violin#spaceX} and where {@link violin#orient}="vertical" + * the main dimension is {@link violin#spaceY} + * @param {boolean} [overflowQ=false] + * @memberof violin# * @property */ - violinPointValueExtractor; - - + overflowQ = true, /** - * Gets / sets the horizontalQ - * (see {@link violin#orient}) - * @param {boolean} [_=none] - * @returns {calculateViolinValues | boolean} - * @memberof calculateViolinValues - * @property - * - * by default horizontalQ = true - */ - calculateViolinValues.horizontalQ = function(_) { return arguments.length ? (horizontalQ=_, calculateViolinValues) : horizontalQ }; + * Whether or not to display points inside the points + * @param {boolean} [pointsQ=false] + * @memberof violin# + * @property + */ + pointsQ = true, /** - * Gets / sets the quartileKeys - * (see {@link violin#quartileKeys}) - * @param {string[]} [_=none] - * @returns {calculateViolinValues | string[]} - * @memberof calculateViolinValues - * @property - * - * by default quartileKeys = ["Q0","Q1","Q2","Q3","Q4"] - */ - calculateViolinValues.quartileKeys = function(_) { return arguments.length ? (quartileKeys=_, calculateViolinValues) : quartileKeys }; + * An array - putatively of other arrays - depicting how bars should be arranged + * @param {Array[]} [grouping=undefined] + * @memberof violin# + * @property + */ + grouping, /** - * Gets / sets the violinPointsExtractor - * (see {@link violin#violinPointsExtractor}) - * @param {function} [_=none] - * @returns {calculateViolinValues | function} - * @memberof calculateViolinValues - * @property - * - * by default violinPointsExtractor = function(violinKey, violinData) { return violinData.points } - */ - calculateViolinValues.violinPointsExtractor = function(_) { return arguments.length ? (violinPointsExtractor=_, calculateViolinValues) : violinPointsExtractor }; + * How to get the value of the violin + * @param {function} [valueExtractor=function(key, index) { return data[key] }] + * @memberof violin# + * @property + */ + valueExtractor = function(key, index) {return data[key] }, /** - * Gets / sets the violinPointValueExtractor - * (see {@link violin#violinPointValueExtractor}) - * @param {function} [_=none] - * @returns {calculateViolinValues | function} - * @memberof calculateViolinValues - * @property - * - * by default violinPointValueExtractor = function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value } - */ - calculateViolinValues.violinPointValueExtractor = function(_) { return arguments.length ? (violinPointValueExtractor=_, calculateViolinValues) : violinPointValueExtractor }; - + * How to sort the bars - if {@link violin#grouping} is not provided. + * @param {function} [sortingFunction=function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}] + * @memberof violin# + * @property + */ + sortingFunction = function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}, /** - * The function produced by neededViolinValues. - * - * Adds the data need to render the violin as an svg - * @param {string} violinKey the key of the current violin - * @param {object} data the object consisting of violinKey - values (violin data) pairs - * @returns {none} this function manipulates the passed data object adding the following keys - * - * data[violinKey].binned // the binned values of the points - * - * data[violinKey].frequencies // the list consisting of the length of each bin - * - * data[violinKey].contour // the points depicting the contour of 1/2 of the violin - * - * data[violinKey].utils.math.quartiles // the utils.math.quartiles of the points' values - * - * data[violinKey].pointKeys // the keys associated with the points - * - * data[violinKey].pointValues // the numerical values of the points - * - * @memberof neededViolinValues# + * The scale for which violin values should be transformed by + * @param {d3.scale} [scale=d3.scaleLinear] + * @memberof violin# * @property */ - function calculateViolinValues(violinKey, data) { - // data for the current violin - var violinData = data[violinKey]; - // the object of points - var violinPoints = violinPointsExtractor(violinKey, violinData); - // - var violinPointsKeys = d3.keys(violinPoints); - // the numerical values of those points - var violinPointsValues = violinPointsKeys.map(function(pk, i){return violinPointValueExtractor(pk, violinPoints)}); - - // utils.math.quartiles of those points - var pointQuartiles = utils.math.quartiles(violinPointsValues, quartileKeys); + scale = d3.scaleLinear(), + /** + * The padding for the domain of the scale (see {@link violin#scale}) + * @param {number} [domainPadding=0.5] + * @memberof violin# + * @property + */ + domainPadding = 0.5, + /** + * Default space for the spacer (percentage) of main dimension given the orientation + * (see {@link violin#orient}), where {@link violin#orient}="horizontal" + * the main dimension is {@link violin#spaceX} and where {@link violin#orient}="vertical" + * the main dimension is {@link violin#spaceY} between bars + * @param {number} [objectSpacer=0.05] + * @memberof violin# + * @property + */ + objectSpacer = 0.05, + /** + * The minimum size that an object can be + * @param {number} [minObjectSize=50] + * @memberof violin# + * @property + */ + minObjectSize = 50, + /** + * The maximum size that an object can be + * @param {number} [maxObjectSize=100] + * @memberof violin# + * @property + */ + maxObjectSize = 100, - // binned points - var binned = d3.histogram()(violinPointsValues); - // length of bins - var frequencies = binned.map(bin=>bin.length); - // min and max countour points for nice drawings - var minContourPoint = horizontalQ ? {x: 0, y: d3.min(violinPointsValues)} : {x: d3.min(violinPointsValues), y: 0}; - var maxContourPoint = horizontalQ ? {x: 0, y: d3.max(violinPointsValues)} : {x: d3.max(violinPointsValues), y: 0}; - var violinContourPoints = binned.map(function(bin, i) { - return horizontalQ - ? {y: (bin.length) ? d3.median(bin): d3.median([bin.x0, bin.x1]), x: frequencies[i]} - : {x: (bin.length) ? d3.median(bin): d3.median([bin.x0, bin.x1]), y: frequencies[i]} - }); - // points along which to draw the violin shpe - violinContourPoints = [minContourPoint].concat(violinContourPoints).concat([maxContourPoint]); - - // set data - violinData.binned = binned; - violinData.frequencies = frequencies; - violinData.contour = violinContourPoints; - violinData.utils.math.quartiles = pointQuartiles; - violinData.pointKeys = violinPointsKeys; - violinData.pointValues = violinPointsValues; - } + /** + * The stroke width of the bars + * @param {number} [barStrokeWidth=2] + * @memberof violin# + * @property + */ + objectStrokeWidth = 2, + /** + * Instance of ColorFunction + * @param {function} [colorFunction = colorFunction()] + * @memberof violin# + * @property + */ + colorFunction$$1 = colorFunction(), + /** + * Instance of ColorFunction modified by a scale for the points + * @param {function} [pointColorFunc = colorFunction()] + * @memberof violin# + * @property + */ + pointColorFunc = function (d, type, base, min, max) { + var minMaxHexScale = d3.scaleLinear().domain([min, max]).range([-0.25, 0.05]); + var scaledColor = utils.color.modifyHexidecimalColorLuminance(base.replace('#', ''), minMaxHexScale(d)); + var mod = type == "stroke" ? 0 : 0.25; + return utils.color.modifyHexidecimalColorLuminance(scaledColor.replace('#', ''), mod) + }, - return calculateViolinValues -} + /** + * The radius of a point + * @param {number} [pointRadius=3] + * @memberof violin# + * @property + */ + pointRadius = 3, + /** + * The stroke width of the oints + * @param {number} [pointStrokeWidth=2] + * @memberof violin# + * @property + */ + pointStrokeWidth = 2, -function upset ( selection ) { - var - data, - orient='horizontal', - spaceX, - spaceY, - overflowQ=false, - minObjectSize=20, - maxObjectSize=50, - circleStrokeWidth=2, - // colorFunction= + /** + * Color of the background + * @param {string} [backgroundFill="transparent"] + * @memberof violin# + * @property + */ backgroundFill = 'transparent', - namespace='d3sm-upset', - objectClass = 'upset', - + /** + * Namespace for all items made by this instance of violin + * @param {string} [namespace="d3sm-violin"] + * @memberof violin# + * @property + */ + namespace = 'd3sm-violin', + /** + * Class name for violin container ( element) + * @param {string} [objectClass="violin"] + * @memberof violin# + * @property + */ + objectClass = 'violin', + /** + * Duration of all transitions of this element + * @param {number} [transitionDuration=1000] + * @memberof violin# + * @property + */ transitionDuration = 1000, + /** + * Easing function for transitions + * @param {d3.ease} [easeFunc=d3.easeExp] + * @memberof violin# + * @property + */ easeFunc = d3.easeExp, - setKey = "set", - intersectionKey = "intersection", - elementsKey = "elements", - - setExtractor = function(key, i) {return data[key][setKey]}, - intersectionExtractor = function(key, i) {return data[key][intersectionKey]}, - elementExtractor = function(key, i) {return data[key][elementsKey]}, - - cellKeys, - setValues, - intersectionValues, - xObjectSpacer = 0.05, - yObjectSpacer = 0.05, - radius, - - // listDelim = ';' - - yObjectSize, - ySpacerSize, - xObjectSize, - xSpacerSize, - - setKeySortingFunction = function(a, b) { return setValues.indexOf(setExtractor(a)) - setValues.indexOf(setExtractor(b)) }, - intersectionKeySortingFunction = function(a, b) { return intersectionValues.indexOf(intersectionExtractor(a)) - intersectionValues.indexOf(intersectionExtractor(b)) }; - - - upset.selection = function(_) { return arguments.length ? (selection = _, upset) : selection; }; - upset.data = function(_) { return arguments.length ? (data = _, upset) : data; }; - upset.orient = function(_) { return arguments.length ? (orient = _, upset) : orient; }; - upset.spaceX = function(_) { return arguments.length ? (spaceX = _, upset) : spaceX; }; - upset.spaceY = function(_) { return arguments.length ? (spaceY = _, upset) : spaceY; }; - upset.overflowQ = function(_) { return arguments.length ? (overflowQ = _, upset) : overflowQ; }; - upset.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, upset) : minObjectSize; }; - upset.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, upset) : maxObjectSize; }; - upset.circleStrokeWidth = function(_) { return arguments.length ? (circleStrokeWidth = _, upset) : circleStrokeWidth; }; - upset.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, upset) : backgroundFill; }; - upset.namespace = function(_) { return arguments.length ? (namespace = _, upset) : namespace; }; - upset.objectClass = function(_) { return arguments.length ? (objectClass = _, upset) : objectClass; }; - upset.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, upset) : transitionDuration; }; - upset.easeFunc = function(_) { return arguments.length ? (easeFunc = _, upset) : easeFunc; }; - upset.cellKeys = function(_) { return arguments.length ? (cellKeys = _, upset) : cellKeys; }; - upset.setValues = function(_) { return arguments.length ? (setValues = _, upset) : setValues; }; - upset.intersectionValues = function(_) { return arguments.length ? (intersectionValues = _, upset) : intersectionValues; }; - upset.xObjectSpacer = function(_) { return arguments.length ? (xObjectSpacer = _, upset) : xObjectSpacer; }; - upset.yObjectSpacer = function(_) { return arguments.length ? (yObjectSpacer = _, upset) : yObjectSpacer; }; - upset.radius = function(_) { return arguments.length ? (radius = _, upset) : radius; }; - upset.setExtractor = function(_) { return arguments.length ? (setExtractor = _, upset) : setExtractor; }; - upset.intersectionExtractor = function(_) { return arguments.length ? (intersectionExtractor = _, upset) : intersectionExtractor; }; - upset.elementExtractor = function(_) { return arguments.length ? (elementExtractor = _, upset) : elementExtractor; }; - upset.setKeySortingFunction = function(_) { return arguments.length ? (setKeySortingFunction = _, upset) : setKeySortingFunction; }; - upset.intersectionKeySortingFunction = function(_) { return arguments.length ? (intersectionKeySortingFunction = _, upset) : intersectionKeySortingFunction; }; - - upset.yObjectSize = function(_) { return arguments.length ? (yObjectSize = _, upset) : yObjectSize; }; - upset.ySpacerSize = function(_) { return arguments.length ? (ySpacerSize = _, upset) : ySpacerSize; }; - upset.xObjectSize = function(_) { return arguments.length ? (xObjectSize = _, upset) : xObjectSize; }; - upset.xSpacerSize = function(_) { return arguments.length ? (xSpacerSize = _, upset) : xSpacerSize; }; - - function upset() { - // for convenience in handling orientation specific values - var horizontalQ = (orient == 'horizontal') ? true : false; - var verticalQ = !horizontalQ; + /** + * The keys corresponding to each quartile + * @param {string[]} [quartileKeys=["Q0", "Q1", "Q2", "Q3", "Q4"]] + * @memberof violin# + * @property + */ + quartileKeys = ["Q0", "Q1", "Q2", "Q3", "Q4"], - // background cliping rectangle - var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; - var container = utils$1.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); + /** + * The keys of the bars + * @param {string[]} [violinKeys=undefined] + * @memberof violin# + * @property + */ + violinKeys, + /** + * The values of the bars + * @param {number[]} [violinValues=undefined] + * @memberof violin# + * @property + */ + violinValues, + /** + * The objectSize (actual width) used by the bars + * @param {number} [objectSize=undefined] + * @memberof violin# + * @property + */ + objectSize, + /** + * The spacerSize (actual width) used by the spacers between the bars + * @param {number} [spacerSize=undefined] + * @memberof violin# + * @property + */ + spacerSize, + /** + * Instance of Tooltip + * @param {function} [tooltip=tooltip()] + * @memberof violin# + * @property + */ + tooltip$$1 = tooltip().keys([quartileKeys[4], quartileKeys[3], quartileKeys[2], quartileKeys[1], quartileKeys[0]]), + pointsTooltip = tooltip(), + // pointKeyExtractor = function(violinKey, violinData, violinValues) {return d3.keys(violinValues[violinKey].values)}, + // pointValueExtractor = function(pointKey, violinKey, violinData, violinValues) {return violinValues[violinKey].values[pointKey]}, - cellKeys = d3.keys(data); - setValues = utils$1.arr.unique(cellKeys.map(setExtractor)).sort(); - intersectionValues = utils$1.arr.unique(cellKeys.map(intersectionExtractor)).sort().sort(function(a, b){ - return a.split(';').length - b.split(';').length - }); - if (!horizontalQ) { - cellKeys.sort(function(a, b){ return setKeySortingFunction(a, b) || intersectionKeySortingFunction(a, b) }); - } else { - cellKeys.sort(function(a, b){ return intersectionKeySortingFunction(a, b) || setKeySortingFunction(a, b) }); - } + /** + * Function which given the key of the violin and that key's associated value + * returns the object consiting of the points of the violin + * (see {@link violin#violinPointsExtractor}) + * @param {Object} [violinPointsExtractor=function(violinKey, violinData) { return violinData.points }] + * @memberof violin# + * @property + */ + violinPointsExtractor = function (violinKey, violinData) {return violinData.points }, + /** + * Function which given the key of the current point and the object of points for the + * violin, returns the numerical value of the point + * (see {@link violin#violinPointValueExtractor}) + * @param {Object} [violinPointValueExtractor=function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value }] + * @memberof violin# + * @property + */ + violinPointValueExtractor = function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value }; + /** + * Gets or sets the violinPointsExtractor + * @param {function} [_=none] + * @returns {violin | function} + * @memberof violin + * @property + * by default violinPointsExtractor = function(violinKey, violinData) { return violinData.points } + */ + violin.violinPointsExtractor = function(_) { return arguments.length ? (violinPointsExtractor = _, violin) : violinPointsExtractor; }; + /** + * Gets or sets the violinPointValueExtractor + * @param {function} [_=none] + * @returns {violin | function} + * @memberof violin + * @property + * by default violinPointsExtractor = function(pointKey, violinKey, violinData, violinValues) {return violinValues[violinKey].values[pointKey]} + */ + violin.violinPointValueExtractor = function(_) { return arguments.length ? (violinPointValueExtractor = _, violin) : violinPointValueExtractor; }; - var - xValues = horizontalQ ? intersectionValues : setValues, - yValues = horizontalQ ? setValues : intersectionValues, - xDim = horizontalQ ? xValues.length : yValues.length, - yDim = horizontalQ ? yValues.length : xValues.length; - - // console.utils.con.log(xValues, yValues) - - - xObjectSize = utils$1.math.calculateWidthOfObject(spaceX, xDim, minObjectSize, maxObjectSize, xObjectSpacer, overflowQ); - yObjectSize = utils$1.math.calculateWidthOfObject(spaceY, yDim, minObjectSize, maxObjectSize, yObjectSpacer, overflowQ); - xSpacerSize = utils$1.math.calculateWidthOfSpacer(xValues, spaceX, xObjectSize, xDim, xObjectSpacer, overflowQ); - ySpacerSize = utils$1.math.calculateWidthOfSpacer(yValues, spaceY, yObjectSize, yDim, yObjectSpacer, overflowQ); - - var ySpacer = groupingSpacer() - .horizontalQ(false) - .moveby('category').numberOfObjects(yDim) - .objectSize(yObjectSize).spacerSize(ySpacerSize) - .transitionDuration(transitionDuration).easeFunc(easeFunc); + /** + * Gets or sets the selection in which items are manipulated + * @param {d3.selection} [_=none] + * @returns {violin | d3.selection} + * @memberof violin + * @property + * by default selection = selection + */ + violin.selection = function(_) { return arguments.length ? (selection = _, violin) : selection; }; + /** + * Gets or sets the data + * (see {@link violin#data}) + * @param {number} [_=none] + * @returns {violin | object} + * @memberof violin + * @property + */ + violin.data = function(_) { return arguments.length ? (data = _, violin) : data; }; + /** + * Gets or sets the orient of the boxes + * (see {@link violin#orient}) + * @param {number} [_=none] + * @returns {violin | object} + * @memberof violin + * @property + */ + violin.orient = function(_) { return arguments.length ? (orient = _, violin) : orient; }; + /** + * Gets or sets the amount of horizontal space in which items are manipulated + * (see {@link violin#spaceX}) + * @param {number} [_=none] should be a number > 0 + * @returns {violin | number} + * @memberof violin + * @property + * by default spaceX = undefined + */ + violin.spaceX = function(_) { return arguments.length ? (spaceX = _, violin) : spaceX; }; + /** + * Gets or sets the amount of vertical space in which items are manipulated + * (see {@link violin#spaceY}) + * @param {number} [_=none] should be a number > 0 + * @returns {violin | number} + * @memberof violin + * @property + * by default spaceY = undefined + */ + violin.spaceY = function(_) { return arguments.length ? (spaceY = _, violin) : spaceY; }; - var xSpacer = groupingSpacer() - .horizontalQ(true) - .moveby('category').numberOfObjects(xDim) - .objectClass(objectClass) - .objectSize(xObjectSize).spacerSize(xSpacerSize) - .transitionDuration(transitionDuration).easeFunc(easeFunc); + /** + * Gets / sets whether or not violin is allowed to go beyond specified dimensions + * (see {@link violin#overflowQ}) + * @param {boolean} [_=none] + * @returns {violin | boolean} + * @memberof violin + * @property + * by default overflowQ = false + */ + violin.overflowQ = function(_) { return arguments.length ? (overflowQ = _, violin) : overflowQ; }; + /** + * Gets / sets whether or not to plot points with the violins + * (see {@link violin#pointsQ}) + * @param {boolean} [_=none] + * @returns {violin | boolean} + * @memberof violin + * @property + * by default pointsQ = false + */ + violin.pointsQ = function(_) { return arguments.length ? (pointsQ = _, violin) : pointsQ; }; - if (verticalQ) { - xSpacer.objectClass(objectClass); - ySpacer.namespace('across').objectClass(utils$1.str.hypenate(objectClass, 'across')); + /** + * Gets / sets the grouping of the boxes + * (see {@link violin#grouping}) + * @param {Array[]} [_=none] + * @returns {violin | Array[]} + * @memberof violin + * @property + * by default grouping = undefined + */ + violin.grouping = function(_) { return arguments.length ? (grouping = _, violin) : grouping; }; + /** + * Gets / sets the valueExtractor + * (see {@link violin#valueExtractor}) + * @param {function} [_=none] + * @returns {violin | function} + * @memberof violin + * @property + */ + violin.valueExtractor = function(_) { return arguments.length ? (valueExtractor = _, violin) : valueExtractor; }; + /** + * Gets / sets the sortingFunction + * (see {@link violin#sortingFunction}) + * @param {function} [_=none] + * @returns {violin | function} + * @memberof violin + * @property + */ + violin.sortingFunction = function(_) { return arguments.length ? (sortingFunction = _, violin) : sortingFunction; }; - ySpacer(container, yValues, 0); - container.selectAll('g.'+utils$1.str.hypenate(objectClass, 'across')) - .each(function(d, i){ xSpacer(d3.select(this), xValues, 0); }); - } else { - xSpacer.namespace('across').objectClass(utils$1.str.hypenate(objectClass, 'across')); - ySpacer.objectClass(objectClass); + /** + * Gets / sets the scale for which the violin values should be transformed by + * (see {@link violin#scale}) + * @param {d3.scale} [_=none] + * @returns {violin | d3.scale} + * @memberof violin + * @property + * by default scale = d3.scaleLinear() + */ + violin.scale = function(_) { return arguments.length ? (scale = _, violin) : scale; }; + /** + * Gets / sets the padding for the domain of the scale + * (see {@link violin#domainPadding}) + * @param {number} [_=none] + * @returns {violin | number} + * @memberof violin + * @property + * by default domainPadding = 0.5 + */ + violin.domainPadding = function(_) { return arguments.length ? (domainPadding = _, violin) : domainPadding; }; - xSpacer(container, xValues, 0); - container.selectAll('g.'+utils$1.str.hypenate(objectClass, 'across')) - .each(function(d, i){ ySpacer(d3.select(this), yValues, 0); }); - } + /** + * Gets / sets objectSpacer + * (see {@link violin#objectSpacer}) + * @param {number} [_=none] + * @returns {violin | number} + * @memberof violin + * @property + * by default objectSpacer = 0.05 + */ + violin.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, violin) : objectSpacer; }; + /** + * Gets / sets the minObjectSize + * (see {@link violin#minObjectSize}) + * @param {number} [_=none] + * @returns {violin | number} + * @memberof violin + * @property + * by default minObjectSize = 15 + */ + violin.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, violin) : minObjectSize; }; + /** + * Gets / sets the maxObjectSize + * (see {@link violin#maxObjectSize}) + * @param {number} [_=none] + * @returns {violin | number} + * @memberof violin + * @property + * by default maxObjectSize = 50 + */ + violin.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, violin) : maxObjectSize; }; - var cells = container.selectAll('g:not(.to-remove).'+objectClass); - var lookup = {}; - cellKeys.map(function(k, i){ - lookup[setExtractor(k)+'::'+intersectionExtractor(k)] = k; - }); + /** + * Gets / sets the objectStrokeWidth + * (see {@link violin#objectStrokeWidth}) + * @param {number} [_=none] + * @returns {violin | number} + * @memberof violin + * @property + * by default objectStrokeWidth = 2 + */ + violin.objectStrokeWidth = function(_) { return arguments.length ? (objectStrokeWidth = _, violin) : objectStrokeWidth; }; - // var positionedCellKeys = [] - // for (var i = 0; i < setValues.length; i++) { - // for (var j = 0; j < intersectionValues.length; j++) { - // var lookupValue = lookup[setValues[j]+"::"+intersectionValues[i]] - // if (lookupValue == undefined) { - // positionedCellKeys.push(undefined) - // } else { - // positionedCellKeys.push(lookupValue) - // } - // console.utils.con.log(i, j, lookupValue) - // } - // } - // - // // console.utils.con.log(positionedCellKeys) - // - // cells.data(positionedCellKeys); + /** + * Gets / sets the colorFunction + * (see {@link violin#colorFunction}) + * @param {colorFunction} [_=none] + * @returns {violin | colorFunction} + * @memberof violin + * @property + * by default colorFunction = colorFunction() + */ + violin.colorFunction = function(_) { return arguments.length ? (colorFunction$$1 = _, violin) : colorFunction$$1; }; + /** + * Gets / sets the colorFunction + * (see {@link violin#colorFunction}) + * @param {colorFunction} [_=none] + * @returns {violin | colorFunction} + * @memberof violin + * @property + * by default colorFunction = colorFunction() + */ + violin.pointColorFunc = function(_) { return arguments.length ? (pointColorFunc = _, violin) : pointColorFunc; }; - cells.data(cellKeys); - cells.each(function(key, i) { - var t = d3.select(this); - if (key == undefined) {return } - var - currentData = data[key]; - // console.utils.con.log(key, currentData) - var - set = setExtractor(key, i), - intersection = intersectionExtractor(key, i); + /** + * Gets / sets the pointRadius + * (see {@link violin#pointRadius}) + * @param {number} [_=none] + * @returns {violin | number} + * @memberof violin + * @property + * by default pointRadius = 2 + */ + violin.pointRadius = function(_) { return arguments.length ? (pointRadius = _, violin) : pointRadius; }; + /** + * Gets / sets the pointStrokeWidth + * (see {@link violin#pointStrokeWidth}) + * @param {number} [_=none] + * @returns {violin | number} + * @memberof violin + * @property + * by default pointStrokeWidth = 2 + */ + violin.pointStrokeWidth = function(_) { return arguments.length ? (pointStrokeWidth = _, violin) : pointStrokeWidth; }; - // console.utils.con.log(set, intersection) - t.classed(intersection, true); - t.classed(set, true); + /** + * Gets / sets the backgroundFill + * (see {@link violin#backgroundFill}) + * @param {string} [_=none] + * @returns {violin | string} + * @memberof violin + * @property + * by default backgroundFill = 'transparent' + */ + violin.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, violin) : backgroundFill; }; + /** + * Gets / sets the namespace + * (see {@link violin#namespace}) + * @param {string} [_=none] + * @returns {violin | string} + * @memberof violin + * @property + * by default namespace = 'd3sm-violin' + */ + violin.namespace = function(_) { return arguments.length ? (namespace = _, violin) : namespace; }; + /** + * Gets / sets the objectClass + * (see {@link violin#objectClass}) + * @param {string} [_=none] + * @returns {violin | string} + * @memberof violin + * @property + * by default objectClass = 'tick-group' + */ + violin.objectClass = function(_) { return arguments.length ? (objectClass = _, violin) : objectClass; }; - var c = utils$1.sel.safeSelect(t, 'circle', utils$1.str.hypenate(objectClass,'circle')); - c.attr('cx', xObjectSize / 2) - .attr('cy', yObjectSize / 2 ) - .attr('r', radius == undefined ? Math.min(xObjectSize, yObjectSize) / 2 : radius) - .attr('fill', intersection.includes(set) ? "black": 'rgb(233,233,233)') - .attr('stroke', "black") - .attr("in-intersection", intersection.includes(set)); - }); + /** + * Gets / sets the transitionDuration + * (see {@link violin#transitionDuration}) + * @param {number} [_=none] + * @returns {violin | number} + * @memberof violin + * @property + * by default transitionDuration = 1000 + */ + violin.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, violin) : transitionDuration; }; + /** + * Gets / sets the easeFunc + * (see {@link violin#easeFunc}) + * @param {d3.ease} [_=none] + * @returns {violin | d3.ease} + * @memberof violin + * @property + * by default easeFunc = d3.easeExp + */ + violin.easeFunc = function(_) { return arguments.length ? (easeFunc = _, violin) : easeFunc; }; - } + /** + * Gets / sets the quartileKey + * (see {@link violin#quartileKey}) + * @param {string} [_=none] + * @returns {violin | string} + * @memberof violin + * @property + * by default quartileKey = "utils.math.quartiles" + */ + violin.quartileKey = function(_) { return arguments.length ? (quartileKey = _, violin) : quartileKey; }; + /** + * Gets / sets the quartileKeys + * (see {@link violin#quartileKeys}) + * @param {string[]} [_=none] + * @returns {violin | string[]} + * @memberof violin + * @property + * by default quartileKeys = ["Q0","Q1","Q2","Q3","Q4"] + */ + violin.quartileKeys = function(_) { return arguments.length ? (quartileKeys = _, violin) : quartileKeys; }; - function intersectionTotals() { - var totals = {}; - // intersectionValues.sort(function(a, b){ return intersectionKeySortingFunction(a, b) }) - intersectionValues.map(function(k, i){ totals[k] = {'total': 0}; }); - cellKeys.map(function(k, i){ - var e = elementExtractor(k, i); - if (totals[intersectionExtractor(k, i)]['total'] == 0) { - if (Array.isArray(e)) { - totals[intersectionExtractor(k, i)]['total']+= e.length; - totals[intersectionExtractor(k, i)]['values'] = e; - } else { - totals[intersectionExtractor(k, i)]['total']+= e; - } + /** + * Gets / sets the violinKeys + * (see {@link violin#violinKeys}) + * @param {string[]} [_=none] + * @returns {violin | string[]} + * @memberof violin + * @property + * by default violinKeys = undefined + */ + violin.violinKeys = function(_) { return arguments.length ? (violinKeys = _, violin) : violinKeys; }; + /** + * Gets / sets the violinValues + * (see {@link violin#violinValues}) + * @param {Object[]} [_=none] + * @returns {violin | Object[]} + * @memberof violin + * @property + * by default violinValues = undefined + */ + violin.violinValues = function(_) { return arguments.length ? (violinValues = _, violin) : violinValues; }; - } - }); - return totals - } + /** + * Gets / sets the objectSize + * (see {@link violin#objectSize}) + * @param {number} [_=none] + * @returns {violin | number} + * @memberof violin + * @property + * by default objectSize = undefined + */ + violin.objectSize = function(_) { return arguments.length ? (objectSize = _, violin) : objectSize; }; + /** + * Gets / sets the spacerSize + * (see {@link violin#spacerSize}) + * @param {number} [_=none] + * @returns {violin | number} + * @memberof violin + * @property + * by default spacerSize = undefined + */ + violin.spacerSize = function(_) { return arguments.length ? (spacerSize = _, violin) : spacerSize; }; + /** + * Gets / sets the tooltip + * (see {@link violin#tooltip}) + * @param {tooltip} [_=none] + * @returns {violin | tooltip} + * @memberof violin + * @property + * by default tooltip = tooltip() + */ + violin.tooltip = function(_) { return arguments.length ? (tooltip$$1 = _, violin) : tooltip$$1; }; + /** + * Gets / sets the pointsTooltip + * (see {@link violin#pointsTooltip}) + * @param {tooltip} [_=none] + * @returns {violin | tooltip} + * @memberof violin + * @property + * by default pointsTooltip = tooltip() + */ + violin.pointsTooltip = function(_) { return arguments.length ? (pointsTooltip = _, violin) : pointsTooltip; }; - function setTotals(){ - var totals = {}; - // intersectionValues.sort(function(a, b){ return intersectionKeySortingFunction(a, b) }) - setValues.map(function(k, i){ totals[k] = {'total': 0}; }); - cellKeys.map(function(k, i){ - var e = elementExtractor(k, i); - if (Array.isArray(e)) { - totals[setExtractor(k, i)]['total']+= e.length; - } else { - totals[setExtractor(k, i)]['total']+= e; - } - }); - return totals - } - upset.intersectionTotals = intersectionTotals; - upset.setTotals = setTotals; - return upset -} + function violin () { + // for convenience in handling orientation specific values + var horizontalQ = (orient == 'horizontal') ? true : false; -let charts = { - scatter, bar, bubble: bubbleHeatmap, heatmap, violin, neededViolinValues, upset -}; + // background cliping rectangle + var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; + var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); -function categoricLegend( selection ) { - var - categories, + // if grouping is undefined sort violinKeys by sortingFunction + var ordered = (grouping == undefined) ? d3.keys(data).sort(sortingFunction) : grouping; - /** - * Data to plot. Assumed to be a object, where each key corresponds to a legend - * (see {@link legend#data}) - * @param {Object} [data=undefined] - * @memberof legend# - * @property - */ - data, - /** - * Which direction to render the bars in - * (see {@link legend#orient}) - * @param {number} [orient='horizontal'] - * @memberof legend# - * @property - */ - orient='horizontal', - /** - * Amount of horizontal space (in pixels) avaible to render the legend in - * (see {@link legend#spaceX}) - * @param {number} [spaceX=undefined] - * @memberof legend# - * @property - */ - spaceX, - /** - * Amount of vertical space (in pixels) avaible to render the legend in - * (see {@link legend.spaceY}) - * @param {number} [spaceY=undefined] - * @memberof legend# - * @property - */ - spaceY, + // console.log(ordered) - /** - * Whether or not to allow legend to render elements pass the main spatial dimension - * given the orientation (see {@link legend#orient}), where {@link legend#orient}="horizontal" - * the main dimension is {@link legend#spaceX} and where {@link legend#orient}="vertical" - * the main dimension is {@link legend#spaceY} - * @param {boolean} [overflowQ=false] - * @memberof legend# - * @property - */ - overflowQ = false, + violinKeys = utils.arr.flatten(ordered); - /** - * An array - putatively of other arrays - depicting how bars should be arranged - * @param {Array[]} [grouping=undefined] - * @memberof legend# - * @property - */ - grouping, - /** - * How to get the value of the legend - * @param {function} [valueExtractor=function(key, index) { return data[key] }] - * @memberof legend# - * @property - */ - valueExtractor = function(key, index) { return data[key] }, - /** - * How to sort the bars - if {@link bar#grouping} is not provided. - * @param {function} [sortingFunction=function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}] - * @memberof bar# - * @property - */ - sortingFunction = function(keyA, keyB) {return d3.ascending(keyA, keyB)}, - /** - * Default space for the spacer (percentage) of main dimension given the orientation - * (see {@link legend#orient}), where {@link legend#orient}="horizontal" - * the main dimension is {@link legend#spaceX} and where {@link legend#orient}="vertical" - * the main dimension is {@link legend#spaceY} between bars - * @param {number} [objectSpacer=0.05] - * @memberof legend# + var calcValues = neededViolinValues() + .horizontalQ(horizontalQ) + .quartileKeys(quartileKeys) + .violinPointsExtractor(violinPointsExtractor) + .violinPointValueExtractor(violinPointValueExtractor); + + + + // augment valus + violinValues = {}; + violinKeys.map(function(vk, i){ + let v = calcValues(vk, data); + violinValues[vk] = v; + }); + + var numberOfObjects = violinKeys.length; + + var min = [].concat(...violinKeys.map(function(k, i){return data[k].quartiles[quartileKeys[0]]})); + var max = [].concat(...violinKeys.map(function(k, i){return data[k].quartiles[quartileKeys[quartileKeys.length - 1]]})); + var extent = [Math.min(...min) - domainPadding, Math.max(...max) + domainPadding]; + // console.log(extent, violinValues, ordered) + + // set the scale + scale.domain(extent).range(horizontalQ ? [0,spaceY] : [0, spaceX]); + var space = horizontalQ ? spaceX : spaceY; + // calculate object size + objectSize = utils.math.calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ); + // calculate spacer size if needed + spacerSize = utils.math.calculateWidthOfSpacer(ordered, space, objectSize, numberOfObjects, objectSpacer, overflowQ); + // make the nested groups + var spacerFunction = groupingSpacer() + .horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects) + .objectClass(objectClass).objectSize(objectSize).spacerSize(spacerSize) + .transitionDuration(transitionDuration).easeFunc(easeFunc) + .namespace(namespace); + + // move stuff + spacerFunction(container, ordered, 0); + // console.log(violinKeys, ordered, container.selectAll('g:not(.to-remove).'+objectClass).nodes()) + + // for color function + var parentIndexArray = []; + container.selectAll('g:not(.to-remove).'+objectClass) + .each(function(d, i){if (utils.arr.hasQ(violinKeys, d)){ parentIndexArray.push(Number(d3.select(this).attr('parent-index')));}}); + + // update color function + colorFunction$$1 = colorFunction$$1.colorBy() == 'index' + ? colorFunction$$1.dataExtent([0, Math.max(...parentIndexArray)]) + : colorFunction$$1.dataExtent(extent); + + /* violiin specific needs */ + + + var frequencyMax = Math.max(...[].concat(...violinKeys.map(function(k, i){return d3.max(data[k].frequencies)}))); + var vScale = d3.scaleLinear().domain([0, frequencyMax]).range([0, objectSize / 2]); + + var lArea = d3.line() + .x(function(d, i){ return horizontalQ ? -vScale(d.x) : scale(d.x)}) + .y(function(d, i){ return horizontalQ ? scale(extent[1]) - scale(d.y) : -vScale(d.y)}) + .curve(d3.curveBasis); + var rArea = d3.line() + .x(function(d, i){ return horizontalQ ? vScale(d.x) : scale(d.x)}) + .y(function(d, i){ return horizontalQ ? scale(extent[1]) - scale(d.y) : vScale(d.y)}) + .curve(d3.curveBasis); + + + + + + + container.selectAll('g:not(.to-remove).'+objectClass).each(function(key, i){ + var t = d3.select(this), + currentData = data[key]; + // needed because bug in selecting .to-remove + if (!utils.arr.hasQ(violinKeys, key)) {return} + var + i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'), + fillColor = colorFunction$$1(key, currentData, i, 'fill'), // prevent duplicate computation + strokeColor = colorFunction$$1(key, currentData, i, 'stroke'), + area = utils.sel.safeSelect(t, 'g', 'area'), + la = utils.sel.safeSelect(area, 'path', 'left'), + ra = utils.sel.safeSelect(area, 'path', 'right'), + quarts = utils.sel.safeSelect(t, 'g', 'quarts'), + lq3 = utils.sel.safeSelect(quarts, 'line', 'q3'), + lq1 = utils.sel.safeSelect(quarts, 'line', 'q1'), + q3 = currentData.quartiles[quartileKeys[3]], + q2 = currentData.quartiles[quartileKeys[2]], + q1 = currentData.quartiles[quartileKeys[1]]; + + t.attr('transform', horizontalQ ? 'translate('+objectSize / 2+',0)' : 'translate(0,'+objectSize / 2+')' ); + // draw curve + la.transition().duration(transitionDuration).attr('d', function(dd, ii){ return lArea(currentData.contour)}) + .attr('fill', fillColor) + .attr('stroke', strokeColor) + .attr('stroke-width', objectStrokeWidth); + + ra.transition().duration(transitionDuration).attr('d', function(dd, ii){ return rArea(currentData.contour)}) + .attr('fill', fillColor) + .attr('stroke', strokeColor) + .attr('stroke-width', objectStrokeWidth); + + area.node().addEventListener('mouseover', function(dd, ii){ + container.selectAll('g.'+objectClass).style('opacity', 0.2); + t.style('opacity', 1); + la.attr('stroke-width',objectStrokeWidth*2); + ra.attr('stroke-width',objectStrokeWidth*2); + }); + area.node().addEventListener('mouseout', function(dd, ii){ + container.selectAll('g.'+objectClass).style('opacity', 1); + la.attr('stroke-width',objectStrokeWidth); + ra.attr('stroke-width',objectStrokeWidth); + }); + + if (pointsQ) { + var ptsContainer = utils.sel.safeSelect(t, 'g', 'points'); + var pts = ptsContainer.selectAll('.point').data(currentData.pointKeys); + pts.on('mouseover', null); + + + var ptsExit = pts.exit().transition().ease(easeFunc).duration(transitionDuration) + .attr('r', 0) + .attr('cy', horizontalQ ? scale(extent[1]) - scale(q2) : vScale(0)) + .attr('cx', horizontalQ ? vScale(0) : scale(q2)).remove(); + + var ptsEnter = pts.enter().append('circle').attr('class', 'point').attr('r', 0) + .attr('cx', horizontalQ ? 0 : scale(q2)) + .attr('cy', horizontalQ ? scale(q2) : 0); + + pts = pts.merge(ptsEnter); + + // console.log(pointsTooltip.header()) + + var pTTips = tooltip().selection(pts).data(violinPointsExtractor(key, currentData)) + .header(pointsTooltip.header()) + .keys(pointsTooltip.keys()) + .values(pointsTooltip.values()); + + pTTips(); + + var pMin = d3.min(currentData.pointValues), pMax = d3.max(currentData.pointValues); + + + + pts.transition().duration(transitionDuration).ease(easeFunc).attr('r', pointRadius) + .attr('cy', function(pointKey, ii){ + var dd = currentData.pointValues[ii]; + if (horizontalQ) { return scale(extent[1]) - scale(dd) } + var j = utils.arr.whichBin(currentData.binned, dd); + var r = Math.random(); + var n = vScale(r * currentData.frequencies[j] * 0.5); + var k = Math.random() > 0.5 ? n : -n; + return k + }) + .attr('cx', function(pointKey, ii){ + var dd = currentData.pointValues[ii]; + if (horizontalQ) { + var j = utils.arr.whichBin(currentData.binned, dd); + var r = Math.random(); + var n = vScale(r * currentData.frequencies[j] * 0.5); + var k = Math.random() > 0.5 ? n : -n; + return k + } + return scale(dd) + }) + .attr('stroke', function(dd, ii) { var dd = currentData.pointValues[ii]; return pointColorFunc(dd, 'stroke', strokeColor, pMin, pMax) }) + .attr('fill' , function(dd, ii) { var dd = currentData.pointValues[ii]; return pointColorFunc(dd, 'fill' , strokeColor, pMin, pMax) }) + .attr('stroke-width', pointStrokeWidth); + + ptsContainer.selectAll('circle.point').on('mouseover', function(dd, ii){ + container.selectAll('g.'+objectClass).style('opacity', 0.2); + t.style('opacity', 1); + la.attr('stroke-width',objectStrokeWidth*2); + ra.attr('stroke-width',objectStrokeWidth*2); + + container.selectAll('.point').style('opacity', 0.2); + d3.select(this).style('opacity', 1).attr('r', pointRadius * 2).attr('stroke-width',pointStrokeWidth*2); + }); + ptsContainer.selectAll('circle.point').on('mouseout', function(dd, ii){ + var e = document.createEvent('SVGEvents'); + e.initEvent('mouseout',true,true); + area.node().dispatchEvent(e); + + container.selectAll('.point').style('opacity', 1); + d3.select(this).attr('stroke-width', pointStrokeWidth).attr('r', pointRadius); + }); + } + else { + cV.selectAll('.point') + .transition().duration(transitionDuration).ease(easeFunc) + .attr('r', 0) + .attr('cy', horizontalQ ? scale(extent[1]) - scale(q2) : vScale(0)) + .attr('cx', horizontalQ ? vScale(0) : scale(q2)) + .remove(); + } + + + }); + + + tooltip$$1.selection(container.selectAll('g:not(.to-remove).'+objectClass + ' .area')); + if (tooltip$$1.data() == undefined) {tooltip$$1.data(data);} + tooltip$$1(); + if (tooltip$$1.values() == undefined) { + tooltip$$1.values([ + function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] }, + function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] }, + function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] }, + function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] }, + function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] } + ]); + + } + + } + + return violin +} + + + + + +/** +* Produces the function which manipulates the violin data to have values needed +* for rendering the violins as svg. +* @returns {function} calculateViolinValues +* @namespace neededViolinValues +*/ +function neededViolinValues() { + var + /** + * Whether or not the orientation of the violins are horizontal + * (see {@link violin#orient}) + * @param {Object} [horizontalQ=true] + * @memberof neededViolinValues# * @property */ - objectSpacer = 0.05, + horizontalQ = true, /** - * The minimum size that an object can be - * @param {number} [minObjectSize=50] - * @memberof legend# + * Keys to be put into the utils.math.quartiles if they need to be calculated. + * (see {@link violin#quartileKeys}) + * @param {Object} [quartileKeys=['Q0', 'Q1', 'Q2', 'Q3', 'Q4']] + * @memberof neededViolinValues# * @property */ - minObjectSize = 10, + quartileKeys = ['Q0', 'Q1', 'Q2', 'Q3', 'Q4'], /** - * The maximum size that an object can be - * @param {number} [maxObjectSize=100] - * @memberof legend# + * Function which given the key of the violin and that key's associated value + * returns the object consiting of the points of the violin + * (see {@link violin#violinPointsExtractor}) + * @param {Object} [violinPointsExtractor=function(violinKey, violinData) { return violinData.points }] + * @memberof neededViolinValues# * @property */ - maxObjectSize = 100, - + violinPointsExtractor, /** - * The stroke width of the bars - * @param {number} [barStrokeWidth=2] - * @memberof legend# + * Function which given the key of the current point and the object of points for the + * violin, returns the numerical value of the point + * (see {@link violin#violinPointValueExtractor}) + * @param {Object} [violinPointValueExtractor=function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value }] + * @memberof neededViolinValues# * @property */ - bubbleStrokeWidth = 2, - /** - * Instance of ColorFunction - * @param {function} [colorFunction = colorFunction()] - * @memberof legend# - * @property - */ - colorFunction$$1 = colorFunction(), + violinPointValueExtractor; + /** - * Color of the background - * @param {string} [backgroundFill="transparent"] - * @memberof legend# - * @property - */ - backgroundFill = 'transparent', + * Gets / sets the horizontalQ + * (see {@link violin#orient}) + * @param {boolean} [_=none] + * @returns {calculateViolinValues | boolean} + * @memberof calculateViolinValues + * @property + * + * by default horizontalQ = true + */ + calculateViolinValues.horizontalQ = function(_) { return arguments.length ? (horizontalQ=_, calculateViolinValues) : horizontalQ }; /** - * Namespace for all items made by this instance of legend - * @param {string} [namespace="d3sm-legend"] - * @memberof legend# - * @property - */ - namespace = 'd3sm-legend', + * Gets / sets the quartileKeys + * (see {@link violin#quartileKeys}) + * @param {string[]} [_=none] + * @returns {calculateViolinValues | string[]} + * @memberof calculateViolinValues + * @property + * + * by default quartileKeys = ["Q0","Q1","Q2","Q3","Q4"] + */ + calculateViolinValues.quartileKeys = function(_) { return arguments.length ? (quartileKeys=_, calculateViolinValues) : quartileKeys }; /** - * Class name for legend container ( element) - * @param {string} [objectClass="legend"] - * @memberof legend# + * Gets / sets the violinPointsExtractor + * (see {@link violin#violinPointsExtractor}) + * @param {function} [_=none] + * @returns {calculateViolinValues | function} + * @memberof calculateViolinValues + * @property + * + * by default violinPointsExtractor = function(violinKey, violinData) { return violinData.points } + */ + calculateViolinValues.violinPointsExtractor = function(_) { return arguments.length ? (violinPointsExtractor=_, calculateViolinValues) : violinPointsExtractor }; + /** + * Gets / sets the violinPointValueExtractor + * (see {@link violin#violinPointValueExtractor}) + * @param {function} [_=none] + * @returns {calculateViolinValues | function} + * @memberof calculateViolinValues + * @property + * + * by default violinPointValueExtractor = function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value } + */ + calculateViolinValues.violinPointValueExtractor = function(_) { return arguments.length ? (violinPointValueExtractor=_, calculateViolinValues) : violinPointValueExtractor }; + + + /** + * The function produced by neededViolinValues. + * + * Adds the data need to render the violin as an svg + * @param {string} violinKey the key of the current violin + * @param {object} data the object consisting of violinKey - values (violin data) pairs + * @returns {none} this function manipulates the passed data object adding the following keys + * + * data[violinKey].binned // the binned values of the points + * + * data[violinKey].frequencies // the list consisting of the length of each bin + * + * data[violinKey].contour // the points depicting the contour of 1/2 of the violin + * + * data[violinKey].utils.math.quartiles // the utils.math.quartiles of the points' values + * + * data[violinKey].pointKeys // the keys associated with the points + * + * data[violinKey].pointValues // the numerical values of the points + * + * @memberof neededViolinValues# * @property */ - objectClass = 'legend', + function calculateViolinValues(violinKey, data) { + // data for the current violin + var violinData = data[violinKey]; + // the object of points + var violinPoints = violinPointsExtractor(violinKey, violinData); + // + var violinPointsKeys = d3.keys(violinPoints); + // the numerical values of those points + var violinPointsValues = violinPointsKeys.map(function(pk, i){return violinPointValueExtractor(pk, violinPoints)}); + + // utils.math.quartiles of those points + var pointQuartiles = utils.math.quartiles(violinPointsValues, quartileKeys); + + // binned points + var binned = d3.histogram()(violinPointsValues); + // length of bins + var frequencies = binned.map(bin=>bin.length); + // min and max countour points for nice drawings + var minContourPoint = horizontalQ ? {x: 0, y: d3.min(violinPointsValues)} : {x: d3.min(violinPointsValues), y: 0}; + var maxContourPoint = horizontalQ ? {x: 0, y: d3.max(violinPointsValues)} : {x: d3.max(violinPointsValues), y: 0}; + var violinContourPoints = binned.map(function(bin, i) { + return horizontalQ + ? {y: (bin.length) ? d3.median(bin): d3.median([bin.x0, bin.x1]), x: frequencies[i]} + : {x: (bin.length) ? d3.median(bin): d3.median([bin.x0, bin.x1]), y: frequencies[i]} + }); + // points along which to draw the violin shpe + violinContourPoints = [minContourPoint].concat(violinContourPoints).concat([maxContourPoint]); + + // set data + violinData.binned = binned; + violinData.frequencies = frequencies; + violinData.contour = violinContourPoints; + violinData.quartiles = pointQuartiles; + violinData.pointKeys = violinPointsKeys; + violinData.pointValues = violinPointsValues; + return violinData + } + + return calculateViolinValues +} + +function upset ( selection ) { + var + data, + orient='horizontal', + spaceX, + spaceY, + overflowQ=false, + minObjectSize=20, + maxObjectSize=50, + circleStrokeWidth=2, + // colorFunction= + backgroundFill = 'transparent', + namespace='d3sm-upset', + objectClass = 'upset', + + transitionDuration = 1000, + easeFunc = d3.easeExp, + + setKey = "set", + intersectionKey = "intersection", + elementsKey = "elements", + + setExtractor = function(key, i) {return data[key][setKey]}, + intersectionExtractor = function(key, i) {return data[key][intersectionKey]}, + elementExtractor = function(key, i) {return data[key][elementsKey]}, + + cellKeys, + setValues, + intersectionValues, + xObjectSpacer = 0.05, + yObjectSpacer = 0.05, + radius, + + // listDelim = ';' + + yObjectSize, + ySpacerSize, + xObjectSize, + xSpacerSize, + + setKeySortingFunction = function(a, b) { return setValues.indexOf(setExtractor(a)) - setValues.indexOf(setExtractor(b)) }, + intersectionKeySortingFunction = function(a, b) { return intersectionValues.indexOf(intersectionExtractor(a)) - intersectionValues.indexOf(intersectionExtractor(b)) }; + + + upset.selection = function(_) { return arguments.length ? (selection = _, upset) : selection; }; + upset.data = function(_) { return arguments.length ? (data = _, upset) : data; }; + upset.orient = function(_) { return arguments.length ? (orient = _, upset) : orient; }; + upset.spaceX = function(_) { return arguments.length ? (spaceX = _, upset) : spaceX; }; + upset.spaceY = function(_) { return arguments.length ? (spaceY = _, upset) : spaceY; }; + upset.overflowQ = function(_) { return arguments.length ? (overflowQ = _, upset) : overflowQ; }; + upset.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, upset) : minObjectSize; }; + upset.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, upset) : maxObjectSize; }; + upset.circleStrokeWidth = function(_) { return arguments.length ? (circleStrokeWidth = _, upset) : circleStrokeWidth; }; + upset.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, upset) : backgroundFill; }; + upset.namespace = function(_) { return arguments.length ? (namespace = _, upset) : namespace; }; + upset.objectClass = function(_) { return arguments.length ? (objectClass = _, upset) : objectClass; }; + upset.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, upset) : transitionDuration; }; + upset.easeFunc = function(_) { return arguments.length ? (easeFunc = _, upset) : easeFunc; }; + upset.cellKeys = function(_) { return arguments.length ? (cellKeys = _, upset) : cellKeys; }; + upset.setValues = function(_) { return arguments.length ? (setValues = _, upset) : setValues; }; + upset.intersectionValues = function(_) { return arguments.length ? (intersectionValues = _, upset) : intersectionValues; }; + upset.xObjectSpacer = function(_) { return arguments.length ? (xObjectSpacer = _, upset) : xObjectSpacer; }; + upset.yObjectSpacer = function(_) { return arguments.length ? (yObjectSpacer = _, upset) : yObjectSpacer; }; + upset.radius = function(_) { return arguments.length ? (radius = _, upset) : radius; }; + upset.setExtractor = function(_) { return arguments.length ? (setExtractor = _, upset) : setExtractor; }; + upset.intersectionExtractor = function(_) { return arguments.length ? (intersectionExtractor = _, upset) : intersectionExtractor; }; + upset.elementExtractor = function(_) { return arguments.length ? (elementExtractor = _, upset) : elementExtractor; }; + upset.setKeySortingFunction = function(_) { return arguments.length ? (setKeySortingFunction = _, upset) : setKeySortingFunction; }; + upset.intersectionKeySortingFunction = function(_) { return arguments.length ? (intersectionKeySortingFunction = _, upset) : intersectionKeySortingFunction; }; + + upset.yObjectSize = function(_) { return arguments.length ? (yObjectSize = _, upset) : yObjectSize; }; + upset.ySpacerSize = function(_) { return arguments.length ? (ySpacerSize = _, upset) : ySpacerSize; }; + upset.xObjectSize = function(_) { return arguments.length ? (xObjectSize = _, upset) : xObjectSize; }; + upset.xSpacerSize = function(_) { return arguments.length ? (xSpacerSize = _, upset) : xSpacerSize; }; + + function upset() { + // for convenience in handling orientation specific values + var horizontalQ = (orient == 'horizontal') ? true : false; + var verticalQ = !horizontalQ; + + // background cliping rectangle + var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; + var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); + + + cellKeys = d3.keys(data); + setValues = utils.arr.unique(cellKeys.map(setExtractor)).sort(); + intersectionValues = utils.arr.unique(cellKeys.map(intersectionExtractor)).sort().sort(function(a, b){ + return a.split(';').length - b.split(';').length + }); + + if (!horizontalQ) { + cellKeys.sort(function(a, b){ return setKeySortingFunction(a, b) || intersectionKeySortingFunction(a, b) }); + } else { + cellKeys.sort(function(a, b){ return intersectionKeySortingFunction(a, b) || setKeySortingFunction(a, b) }); + } + + + + + var + xValues = horizontalQ ? intersectionValues : setValues, + yValues = horizontalQ ? setValues : intersectionValues, + xDim = horizontalQ ? xValues.length : yValues.length, + yDim = horizontalQ ? yValues.length : xValues.length; + + // console.utils.con.log(xValues, yValues) + + + xObjectSize = utils.math.calculateWidthOfObject(spaceX, xDim, minObjectSize, maxObjectSize, xObjectSpacer, overflowQ); + yObjectSize = utils.math.calculateWidthOfObject(spaceY, yDim, minObjectSize, maxObjectSize, yObjectSpacer, overflowQ); + xSpacerSize = utils.math.calculateWidthOfSpacer(xValues, spaceX, xObjectSize, xDim, xObjectSpacer, overflowQ); + ySpacerSize = utils.math.calculateWidthOfSpacer(yValues, spaceY, yObjectSize, yDim, yObjectSpacer, overflowQ); + + var ySpacer = groupingSpacer() + .horizontalQ(false) + .moveby('category').numberOfObjects(yDim) + .objectSize(yObjectSize).spacerSize(ySpacerSize) + .transitionDuration(transitionDuration).easeFunc(easeFunc); + + var xSpacer = groupingSpacer() + .horizontalQ(true) + .moveby('category').numberOfObjects(xDim) + .objectClass(objectClass) + .objectSize(xObjectSize).spacerSize(xSpacerSize) + .transitionDuration(transitionDuration).easeFunc(easeFunc); + + + + if (verticalQ) { + xSpacer.objectClass(objectClass); + ySpacer.namespace('across').objectClass(utils.str.hypenate(objectClass, 'across')); + + ySpacer(container, yValues, 0); + container.selectAll('g.'+utils.str.hypenate(objectClass, 'across')) + .each(function(d, i){ xSpacer(d3.select(this), xValues, 0); }); + } else { + xSpacer.namespace('across').objectClass(utils.str.hypenate(objectClass, 'across')); + ySpacer.objectClass(objectClass); + + xSpacer(container, xValues, 0); + container.selectAll('g.'+utils.str.hypenate(objectClass, 'across')) + .each(function(d, i){ ySpacer(d3.select(this), yValues, 0); }); + } + + + var cells = container.selectAll('g:not(.to-remove).'+objectClass); + var lookup = {}; + cellKeys.map(function(k, i){ + lookup[setExtractor(k)+'::'+intersectionExtractor(k)] = k; + }); + + // var positionedCellKeys = [] + // for (var i = 0; i < setValues.length; i++) { + // for (var j = 0; j < intersectionValues.length; j++) { + // var lookupValue = lookup[setValues[j]+"::"+intersectionValues[i]] + // if (lookupValue == undefined) { + // positionedCellKeys.push(undefined) + // } else { + // positionedCellKeys.push(lookupValue) + // } + // console.utils.con.log(i, j, lookupValue) + // } + // } + // + // // console.utils.con.log(positionedCellKeys) + // + // cells.data(positionedCellKeys); + + + cells.data(cellKeys); + + cells.each(function(key, i) { + var t = d3.select(this); + if (key == undefined) {return } + var + currentData = data[key]; + // console.utils.con.log(key, currentData) + var + set = setExtractor(key, i), + intersection = intersectionExtractor(key, i); + + // console.utils.con.log(set, intersection) + + t.classed(intersection, true); + t.classed(set, true); + + var c = utils.sel.safeSelect(t, 'circle', utils.str.hypenate(objectClass,'circle')); + c.attr('cx', xObjectSize / 2) + .attr('cy', yObjectSize / 2 ) + .attr('r', radius == undefined ? Math.min(xObjectSize, yObjectSize) / 2 : radius) + .attr('fill', intersection.includes(set) ? "black": 'rgb(233,233,233)') + .attr('stroke', "black") + .attr("in-intersection", intersection.includes(set)); + }); + + + + } + + function intersectionTotals() { + var totals = {}; + // intersectionValues.sort(function(a, b){ return intersectionKeySortingFunction(a, b) }) + + intersectionValues.map(function(k, i){ totals[k] = {'total': 0}; }); + cellKeys.map(function(k, i){ + var e = elementExtractor(k, i); + if (totals[intersectionExtractor(k, i)]['total'] == 0) { + if (Array.isArray(e)) { + totals[intersectionExtractor(k, i)]['total']+= e.length; + totals[intersectionExtractor(k, i)]['values'] = e; + } else { + totals[intersectionExtractor(k, i)]['total']+= e; + } + + } + }); + return totals + } + + function setTotals(){ + var totals = {}; + // intersectionValues.sort(function(a, b){ return intersectionKeySortingFunction(a, b) }) + + setValues.map(function(k, i){ totals[k] = {'total': 0}; }); + + cellKeys.map(function(k, i){ + var e = elementExtractor(k, i); + if (Array.isArray(e)) { + totals[setExtractor(k, i)]['total']+= e.length; + } else { + totals[setExtractor(k, i)]['total']+= e; + } + }); + return totals + } + + upset.intersectionTotals = intersectionTotals; + upset.setTotals = setTotals; + + return upset +} + +let charts = { + scatter, bar, bubble: bubbleHeatmap, boxwhisker, heatmap, violin, neededViolinValues, upset +}; + +function categoricLegend( selection ) { + var + categories, + + /** + * Data to plot. Assumed to be a object, where each key corresponds to a legend + * (see {@link legend#data}) + * @param {Object} [data=undefined] + * @memberof legend# + * @property + */ + data, + /** + * Which direction to render the bars in + * (see {@link legend#orient}) + * @param {number} [orient='horizontal'] + * @memberof legend# + * @property + */ + orient='horizontal', + /** + * Amount of horizontal space (in pixels) avaible to render the legend in + * (see {@link legend#spaceX}) + * @param {number} [spaceX=undefined] + * @memberof legend# + * @property + */ + spaceX, + /** + * Amount of vertical space (in pixels) avaible to render the legend in + * (see {@link legend.spaceY}) + * @param {number} [spaceY=undefined] + * @memberof legend# + * @property + */ + spaceY, + + /** + * Whether or not to allow legend to render elements pass the main spatial dimension + * given the orientation (see {@link legend#orient}), where {@link legend#orient}="horizontal" + * the main dimension is {@link legend#spaceX} and where {@link legend#orient}="vertical" + * the main dimension is {@link legend#spaceY} + * @param {boolean} [overflowQ=false] + * @memberof legend# + * @property + */ + overflowQ = false, + + /** + * An array - putatively of other arrays - depicting how bars should be arranged + * @param {Array[]} [grouping=undefined] + * @memberof legend# + * @property + */ + grouping, + + /** + * How to get the value of the legend + * @param {function} [valueExtractor=function(key, index) { return data[key] }] + * @memberof legend# + * @property + */ + valueExtractor = function(key, index) { return data[key] }, + /** + * How to sort the bars - if {@link bar#grouping} is not provided. + * @param {function} [sortingFunction=function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}] + * @memberof bar# + * @property + */ + sortingFunction = function(keyA, keyB) {return d3.ascending(keyA, keyB)}, + /** + * Default space for the spacer (percentage) of main dimension given the orientation + * (see {@link legend#orient}), where {@link legend#orient}="horizontal" + * the main dimension is {@link legend#spaceX} and where {@link legend#orient}="vertical" + * the main dimension is {@link legend#spaceY} between bars + * @param {number} [objectSpacer=0.05] + * @memberof legend# + * @property + */ + objectSpacer = 0.05, + /** + * The minimum size that an object can be + * @param {number} [minObjectSize=50] + * @memberof legend# + * @property + */ + minObjectSize = 10, + /** + * The maximum size that an object can be + * @param {number} [maxObjectSize=100] + * @memberof legend# + * @property + */ + maxObjectSize = 100, + + /** + * The stroke width of the bars + * @param {number} [barStrokeWidth=2] + * @memberof legend# + * @property + */ + bubbleStrokeWidth = 2, + /** + * Instance of ColorFunction + * @param {function} [colorFunction = colorFunction()] + * @memberof legend# + * @property + */ + colorFunction$$1 = colorFunction(), + + /** + * Color of the background + * @param {string} [backgroundFill="transparent"] + * @memberof legend# + * @property + */ + backgroundFill = 'transparent', + /** + * Namespace for all items made by this instance of legend + * @param {string} [namespace="d3sm-legend"] + * @memberof legend# + * @property + */ + namespace = 'd3sm-legend', + /** + * Class name for legend container ( element) + * @param {string} [objectClass="legend"] + * @memberof legend# + * @property + */ + objectClass = 'legend', + + /** + * Duration of all transitions of this element + * @param {number} [transitionDuration=1000] + * @memberof legend# + * @property + */ + transitionDuration = 1000, + /** + * Easing function for transitions + * @param {d3.ease} [easeFunc=d3.easeExp] + * @memberof legend# + * @property + */ + easeFunc = d3.easeExp; + + + legend.categories = function(_) { return arguments.length ? (categories=_, legend) : categories }; + + /** + * Gets or sets the selection in which items are manipulated + * @param {d3.selection} [_=none] + * @returns {legend | d3.selection} + * @memberof legend + * @property + * by default selection = selection + */ + legend.selection = function(_) { return arguments.length ? (selection = _, legend) : selection; }; + /** + * Gets or sets the data + * (see {@link legend#data}) + * @param {number} [_=none] + * @returns {legend | object} + * @memberof legend + * @property + */ + legend.data = function(_) { return arguments.length ? (data = _, legend) : data; }; + /** + * Gets or sets the orient of the bars + * (see {@link legend#orient}) + * @param {number} [_=none] + * @returns {legend | object} + * @memberof legend + * @property + */ + legend.orient = function(_) { return arguments.length ? (orient = _, legend) : orient; }; + /** + * Gets or sets the amount of horizontal space in which items are manipulated + * (see {@link legend#spaceX}) + * @param {number} [_=none] should be a number > 0 + * @returns {legend | number} + * @memberof legend + * @property + * by default spaceX = undefined + */ + legend.spaceX = function(_) { return arguments.length ? (spaceX = _, legend) : spaceX; }; + /** + * Gets or sets the amount of vertical space in which items are manipulated + * (see {@link legend#spaceY}) + * @param {number} [_=none] should be a number > 0 + * @returns {legend | number} + * @memberof legend + * @property + * by default spaceY = undefined + */ + legend.spaceY = function(_) { return arguments.length ? (spaceY = _, legend) : spaceY; }; + + /** + * Gets / sets whether or not legend is allowed to go beyond specified dimensions + * (see {@link legend#spaceX}) + * @param {boolean} [_=none] + * @returns {legend | boolean} + * @memberof legend + * @property + * by default overflowQ = false + */ + legend.overflowQ = function(_) { return arguments.length ? (overflowQ = _, legend) : overflowQ; }; + /** + * Gets / sets the grouping of the bars + * (see {@link legend#grouping}) + * @param {Array[]} [_=none] + * @returns {legend | Array[]} + * @memberof legend + * @property + * by default grouping = undefined + */ + legend.grouping = function(_) { return arguments.length ? (grouping = _, legend) : grouping; }; + /** + * Gets / sets the valueExtractor + * (see {@link legend#valueExtractor}) + * @param {function} [_=none] + * @returns {legend | function} + * @memberof legend + * @property + * by default valueExtractor = function(key, index) { return data[key] }, + */ + legend.valueExtractor = function(_) { return arguments.length ? (valueExtractor = _, legend) : valueExtractor; }; + /** + * Gets / sets the sortingFunction + * (see {@link bar#sortingFunction}) + * @param {function} [_=none] + * @returns {bar | function} + * @memberof bar + * @property + * by default sortingFunction = function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}, + */ + legend.sortingFunction = function(_) { return arguments.length ? (sortingFunction = _, legend) : sortingFunction; }; + /** + * Gets / sets objectSpacer + * (see {@link legend#objectSpacer}) + * @param {number} [_=none] + * @returns {legend | number} + * @memberof legend + * @property + * by default objectSpacer = 0.05 + */ + legend.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, legend) : objectSpacer; }; + /** + * Gets / sets the minObjectSize + * (see {@link legend#minObjectSize}) + * @param {number} [_=none] + * @returns {legend | number} + * @memberof legend + * @property + * by default minObjectSize = 50 + */ + legend.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, legend) : minObjectSize; }; + /** + * Gets / sets the maxObjectSize + * (see {@link legend#maxObjectSize}) + * @param {number} [_=none] + * @returns {legend | number} + * @memberof legend + * @property + * by default maxObjectSize = 100 + */ + legend.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, legend) : maxObjectSize; }; + + /** + * Gets / sets the barStrokeWidth + * (see {@link legend#barStrokeWidth}) + * @param {number} [_=none] + * @returns {legend | number} + * @memberof legend + * @property + * by default barStrokeWidth = 2 + */ + legend.bubbleStrokeWidth = function(_) { return arguments.length ? (bubbleStrokeWidth = _, legend) : bubbleStrokeWidth; }; + /** + * Gets / sets the colorFunction + * (see {@link legend#colorFunction}) + * @param {number} [_=none] + * @returns {legend | number} + * @memberof legend + * @property + * by default colorFunction = colorFunction() + */ + legend.colorFunction = function(_) { return arguments.length ? (colorFunction$$1 = _, legend) : colorFunction$$1; }; + + /** + * Gets / sets the backgroundFill + * (see {@link legend#backgroundFill}) + * @param {string} [_=none] + * @returns {legend | string} + * @memberof legend + * @property + * by default backgroundFill = 'transparent' + */ + legend.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, legend) : backgroundFill; }; + /** + * Gets / sets the namespace + * (see {@link legend#namespace}) + * @param {string} [_=none] + * @returns {legend | string} + * @memberof legend + * @property + * by default namespace = 'd3sm-legend' + */ + legend.namespace = function(_) { return arguments.length ? (namespace = _, legend) : namespace; }; + /** + * Gets / sets the objectClass + * (see {@link legend#objectClass}) + * @param {string} [_=none] + * @returns {legend | string} + * @memberof legend + * @property + * by default objectClass = 'tick-group' + */ + legend.objectClass = function(_) { return arguments.length ? (objectClass = _, legend) : objectClass; }; + /** + * Gets / sets the transitionDuration + * (see {@link legend#transitionDuration}) + * @param {number} [_=none] + * @returns {legend | number} + * @memberof legend + * @property + * by default transitionDuration = 1000 + */ + legend.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, legend) : transitionDuration; }; + /** + * Gets / sets the easeFunc + * (see {@link legend#easeFunc}) + * @param {d3.ease} [_=none] + * @returns {legend | d3.ease} + * @memberof legend + * @property + * by default easeFunc = d3.easeExp + */ + legend.easeFunc = function(_) { return arguments.length ? (easeFunc = _, legend) : easeFunc; }; + + + function legend() { + var horizontalQ = (orient == 'horizontal') ? true : false; + var verticalQ = !horizontalQ; + // background cliping rectangle + var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; + var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); + + + colorFunction$$1.dataExtent([0, categories.length - 1]) + .colorBy('categories') + .categoryExtractor(function(k, v, i){return v}); + + var r = Math.min(spaceX, spaceY) / 2; + var numberOfObjects = categories.length; + + // if grouping is undefined sort barKeys by sortingFunction + var ordered = (grouping == undefined) ? categories.sort(sortingFunction) : grouping; + // ordered might be nested depending on grouping + var catKeys = utils.arr.flatten(ordered); + + var space = horizontalQ ? spaceX : spaceY; + // calculate object size + var objectSize = utils.math.calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ); + // calculate spacer size if needed + var spacerSize = utils.math.calculateWidthOfSpacer(catKeys, space, objectSize, numberOfObjects, objectSpacer, overflowQ); + // make the nested groups + var spacerFunction = groupingSpacer() + .horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects) + .objectClass(objectClass).objectSize(objectSize).spacerSize(spacerSize) + .transitionDuration(transitionDuration).easeFunc(easeFunc) + .namespace(namespace); + + spacerFunction(container, ordered, 0); + var r = Math.min(objectSize, spaceX, spaceY) / 2 - bubbleStrokeWidth; + + container.selectAll('g:not(.to-remove).'+objectClass).each(function(cat, i) { + var t = d3.select(this); + var c = utils.sel.safeSelect(t, 'circle'); + var fillColor = colorFunction$$1(undefined, cat, i, 'fill'), // prevent duplicate computation + strokeColor = colorFunction$$1(undefined, cat, i, 'stroke'); + + var cx = horizontalQ + ? r+bubbleStrokeWidth + : (spaceX - r*2) / 2 + r; + var cy = verticalQ + ? r+bubbleStrokeWidth + : (spaceX - r*2) / 2 + r; + + c.attr("r", r) + .attr('cx', cx) + .attr('cy', cy) + .attr('fill', fillColor) + .attr('stroke', strokeColor) + .attr('stroke-width', bubbleStrokeWidth); + + var text = utils.sel.safeSelect(t, 'text'); + text.text(cat) + .attr('text-anchor', 'middle') + .attr("transform", function(d, i){ + var + x = cx, + y = cy + text.node().getBoundingClientRect().height / 4, + t = 'translate('+x+','+y+')'; + return t + }); + + }); + + + } + + return legend +} + +function numericLegend( selection ) { + + var + min=0, + max=1, + spaceX, + spaceY, + colorFunction$$1 = colorFunction(), + namespace='d3sm-linear-vertical-gradient', + fontSize = 12, + backgroundFill = 'transparent', + textColor = 'black', + roundTo = 2; + + + legend.min = function(_) { return arguments.length ? (min=_, legend) : min }; + legend.max = function(_) { return arguments.length ? (max=_, legend) : max }; + legend.spaceX = function(_) { return arguments.length ? (spaceX=_, legend) : spaceX }; + legend.spaceY = function(_) { return arguments.length ? (spaceY=_, legend) : spaceY }; + legend.namespace = function(_) { return arguments.length ? (namespace=_, legend) : namespace }; + legend.fontSize = function(_) { return arguments.length ? (fontSize=_, legend) : fontSize }; + legend.backgroundFill = function(_) { return arguments.length ? (backgroundFill=_, legend) : backgroundFill }; + legend.colorFunction = function(_) { return arguments.length ? (colorFunction$$1=_, legend) : colorFunction$$1 }; + legend.textColor = function(_) { return arguments.length ? (textColor=_, legend) : textColor }; + legend.roundTo = function(_) { return arguments.length ? (roundTo=_, legend) : roundTo }; + + function legend() { + // background cliping rectangle + var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; + var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); + + var defs = utils.sel.safeSelect(selection, 'defs'); + var linearGradient = utils.sel.safeSelect(defs, 'linearGradient') + .attr("x1", "0%") + .attr("y1", "100%") + .attr("x2", "0%") + .attr("y2", "0%") + .attr('id', utils.str.hypenate(namespace,'numerical-legend-gradient')); + + + colorFunction$$1.dataExtent([min, max]) + .colorBy('value') + .valueExtractor(function(k, v, i){return v}); + + + linearGradient.selectAll('stop') + .data( colorFunction$$1.colors() ) + .enter() + .append('stop') + .attr("offset", function(d, i){ return i / (colorFunction$$1.colors().length - 1) }) + .attr('stop-color', function(d) {return d}); + + + + + var rect = utils.sel.safeSelect(container, 'rect', 'legend') + .attr('transform', 'translate(0,'+fontSize+')') + .style("fill", "url(#"+utils.str.hypenate(namespace,'numerical-legend-gradient')+")") + .attr('x', 0) + .attr('y', 0) + .attr('width', spaceX) + .attr('height', spaceY - fontSize*2) + .on('mousemove', function(d, i){legendMousemove(d, i, rect, d3.select(this));}) + .on('mouseout', function(d, i){ d3.select("#"+utils.str.hypenate(namespace,'legend-tooltip')).remove(); }); + + var minText = utils.sel.safeSelect(container, 'text', 'min') + .text(utils.math.round(min, 2)) + .attr('text-anchor', 'middle') + .attr("font-size", fontSize+'px') + .attr('transform', function(d, i){ + var + x = spaceX / 2, + y = spaceY - fontSize / 4, + t = 'translate('+x+','+y+')'; + return t + }); + + var maxText = utils.sel.safeSelect(container, 'text', 'max') + .text(utils.math.round(max, 2)) + .attr('text-anchor', 'middle') + .attr("font-size", fontSize+'px') + .attr('transform', function(d, i){ + var + x = spaceX / 2, + y = fontSize, + t = 'translate('+x+','+y+')'; + return t + }); + + + + + } + + function legendMousemove(d, i, rect, t) { + var s = d3.scaleLinear() + .domain([0, rect.attr('height')]) + .range([max, min]); + var m = d3.mouse(rect.node()); + var v = utils.math.round(s(m[1]),roundTo); + + var strokeColor = colorFunction$$1(undefined, v, undefined, 'stroke'); + var fillColor = colorFunction$$1(undefined, v, undefined, 'fill'); + + var div = utils.sel.safeSelect(d3.select('body'), 'div', utils.str.hypenate(namespace,'legend-tooltip')) + .attr('id', utils.str.hypenate(namespace,'legend-tooltip')) + .style('position', 'absolute') + .style('left', (d3.event.pageX+15)+'px') + .style('top', (d3.event.pageY+15)+'px') + .style('background-color', fillColor) + .style('border-color', strokeColor) + + .style('min-width', (fontSize * (String(max).split('.')[0].length+3))+'px') + .style('min-height', (fontSize * (String(max).split('.')[0].length+3))+'px') + .style('border-radius', '50%') + .style('border-radius', '5000px') + + .style('display', 'flex') + .style('justify-content', 'center') + .style('text-align', 'middle') + .style('padding', 2+"px") + + .style('border-style', 'solid') + .style('border-width', 2); + + var text = utils.sel.safeSelect(div, 'div') + .text(v) + .style('color', textColor) + .style('align-self', 'center'); + } + + return legend +} + +let legends = { + categorical: categoricLegend, numeric: numericLegend +}; + +// import * as d3 from "d3"; +/******************************************************************************* +** ** +** ** +** D3 EXTENSIONS ** +** ** +** ** +*******************************************************************************/ +/** +* Recursively ascends parents of selection until it finds an svg tag +* @function d3.selection.thisSVG +* @augments d3.selection +* @returns {Element} which is the svg tag, not the d3 selection of that tag +*/ +d3.selection.prototype.thisSVG = function() { return utils.sel.getContainingSVG(this.node()); }; + + +/** +* Helper for getting absolute position of the mouse +* @function d3.mouse.absolute +* @augments d3.mouse +* @returns {number[]} [x, y] as they relate to `html` not to local scope. +*/ +d3.mouse.absolute = function() { + var html = d3.select('html').node(); + var [x, y] = this(html); + return [x, y] +}; + + +/** +* Gets position of the selection in relation to the containing svg +* @see{@link getContainingSVG} +* @function d3.selection.absolutePosition +* @augments d3.selection +* @returns {Object} with structure similar to getBoundingClientRect, e.g. +* top, left, bottom, right, height, width +*/ +d3.selection.prototype.absolutePosition = function() { + var element = this.node(); + var elementPosition = element.getBoundingClientRect(); + var containerSVG = utils.sel.getContainingSVG(element); + var svgPosition = containerSVG.getBoundingClientRect(); + + return { + top: elementPosition.top - svgPosition.top, + left: elementPosition.left - svgPosition.left, + bottom: elementPosition.bottom - svgPosition.top, + right: elementPosition.right - svgPosition.left, + height: elementPosition.height, + width: elementPosition.width + }; + +}; + + +d3.selection.prototype.relativePositionTo = function(container) { + var element = this.node(); + var elementPosition = element.getBoundingClientRect(); + var containerSVG = container; + var svgPosition = containerSVG.getBoundingClientRect(); + + return { + top: elementPosition.top - svgPosition.top, + left: elementPosition.left - svgPosition.left, + bottom: elementPosition.bottom - svgPosition.top, + right: elementPosition.right - svgPosition.left, + height: elementPosition.height, + width: elementPosition.width + }; + +}; + +function getTranslation$1(selection){ + var transform = selection.attr('transform'); + var [junk, xy] =transform.split('translate('); + var [x, y] = xy.split(','); + junk = y.split(')'); + return [parseFloat(x), parseFloat(y)] +} + +function lasso( selection ) { + var + svg, // svg that is target of events + objectContainer, // container which houses objects we are selecting (allows for transform to be applied to lasso) + objectClass, // class of object we are selecting + namespace="d3sm-lasso", + chartContainer, + chartOffset, + objectsOffset, + eventCatcher, + + xScale, // optional scale for the lasso currentPoints + yScale, // optional scale for the lasso currentPoints + + activeQ = false, // whether or not lasso is active + + currentPoints=[], // mouse points for current lasso + allPoints=[], // list of lists for all points of lassos + + line = d3.line() + .x(function(d, i){ + var x; + if (xScale != undefined) { x = xScale(d[0]); } + else {x = d[0];} + return x //- chartOffset[0]// - objectsOffset[0] + }) + .y(function(d, i){ + var y; + if (yScale != undefined) { y= yScale(d[1]); } + else {y = d[1];} + return y// - chartOffset[1]// - objectsOffset[1] + }) + .curve(d3.curveLinearClosed), + + instance=0, // an indentifier for which instance this lasso is under the current svg + + tickDistance = 10, + + // styles for lasso path + color$$1 = '#17a2b8', + animationRate = '10s', + opacity=0.3, + dashArray = '5, 10', + stroke = 'black', + strokeWidth=2, + + // styles for lassoed objects + lassoedFill = "white", + lassoedStroke = 'black', + lassoedStrokeWidth = 3, + + transitionDuration = 1000, + easeFunc = d3.easeExp; + + var path; + + lasso.svg = function(_) { return arguments.length ? (svg = _, lasso) : svg; }; + lasso.chartContainer = function(_) { return arguments.length ? (chartContainer = _, lasso) : chartContainer; }; + lasso.objectContainer = function(_) { return arguments.length ? (objectContainer = _, lasso) : objectContainer; }; + lasso.objectClass = function(_) { return arguments.length ? (objectClass = _, lasso) : objectClass; }; + lasso.namespace = function(_) { return arguments.length ? (namespace = _, lasso) : namespace; }; + lasso.xScale = function(_) { return arguments.length ? (xScale = _, lasso) : xScale; }; + lasso.yScale = function(_) { return arguments.length ? (yScale = _, lasso) : yScale; }; + lasso.activeQ = function(_) { return arguments.length ? (activeQ = _, lasso) : activeQ; }; + lasso.currentPoints = function(_) { return arguments.length ? (currentPoints = _, lasso) : currentPoints; }; + lasso.allPoints = function(_) { return arguments.length ? (allPoints = _, lasso) : allPoints; }; + lasso.instance = function(_) { return arguments.length ? (instance = _, lasso) : instance; }; + lasso.tickDistance = function(_) { return arguments.length ? (tickDistance = _, lasso) : tickDistance; }; + lasso.color = function(_) { return arguments.length ? (color$$1 = _, lasso) : color$$1; }; + lasso.animationRate = function(_) { return arguments.length ? (animationRate = _, lasso) : animationRate; }; + lasso.opacity = function(_) { return arguments.length ? (opacity = _, lasso) : opacity; }; + lasso.dashArray = function(_) { return arguments.length ? (dashArray = _, lasso) : dashArray; }; + lasso.stroke = function(_) { return arguments.length ? (stroke = _, lasso) : stroke; }; + lasso.lassoedFill = function(_) { return arguments.length ? (lassoedFill = _, lasso) : lassoedFill; }; + lasso.lassoedStroke = function(_) { return arguments.length ? (lassoedStroke = _, lasso) : lassoedStroke; }; + lasso.lassoedStrokeWidth = function(_) { return arguments.length ? (lassoedStrokeWidth = _, lasso) : lassoedStrokeWidth; }; + lasso.eventCatcher = function(_) { return arguments.length ? (eventCatcher = _, lasso) : eventCatcher; }; + + lasso.drag = drag; + lasso.draw = draw; + lasso.tick = tick; + lasso.detect = detect; + lasso.toggle = toggle; + lasso.remove = remove; + lasso.render = render; + lasso.keyFrames = keyFrames; + lasso.updateObjects = updateObjects; + lasso.applyPathAttributes = applyPathAttributes; + lasso.applyObjectAttributes = applyObjectAttributes; + + keyFrames(); + + function lasso() { + // add a dash animation if needed + if (activeQ) { transitionDraw(); } + } + + function toggle(state) { + // use optional param to set state, otherwise toggle state + activeQ = (state!=undefined) ? state : !activeQ; + chartOffset = getTranslation$1(chartContainer); //utils.sel.getTranslation(chartContainer) + objectsOffset = getTranslation$1(objectContainer); //utils.sel.getTranslation(objectContainer) + + if (activeQ) { + svg.node().addEventListener('mousedown', render, true); + } else { + svg.node().removeEventListener('mousedown', render, true); + remove(); + } + + } + + function draw() { + chartOffset = getTranslation$1(chartContainer); //utils.sel.getTranslation(chartContainer) + objectsOffset = getTranslation$1(objectContainer); //utils.sel.getTranslation(objectContainer) + + var container = utils.sel.safeSelect(objectContainer, 'g', 'lasso-container'); + var paths$$1 = container.selectAll('path[instance="'+instance+'"]'); + + // update + paths$$1 = paths$$1.data(allPoints); + + // remove excess + var pExit = paths$$1.exit().remove(); + // add needed paths + var pEnter = paths$$1.enter().append('path'); + + // merge + paths$$1 = paths$$1.merge(pEnter); + + // apply + applyPathAttributes(paths$$1); + } + + function remove() { + var container = utils.sel.safeSelect(objectContainer, 'g', 'lasso-container'); + var paths$$1 = container.selectAll('path[instance="'+instance+'"]').remove(); + container.remove(); + objectContainer.selectAll(objectClass).classed("in-lasso", false); + updateObjects(); + } + + function render( event ) { + // nothing can interefer with drawing the lasso + event.preventDefault(); event.stopPropagation(); + + var container = utils.sel.safeSelect(objectContainer, 'g', 'lasso-container'); + + /* + each time the user presses down, while the state is active, the lasso + the lasso should make a seperate segment. + */ + currentPoints = []; + + svg.node().addEventListener('mousemove', drag); + svg.node().addEventListener('mouseup', function(event) { + svg.node().removeEventListener('mousemove', drag); + allPoints.push(currentPoints); + // BUG: somehow this is pushing currentPoints n times where n is the nth lasso path for the current instance + // NOTE: allPoints = utils.arr.unique(allPoints) is a temporary and inefficient fix + allPoints = utils.arr.unique(allPoints); + }); + + path = container.append('path').data([currentPoints]); + applyPathAttributes(path); + } + + function transitionDraw() { + var container = utils.sel.safeSelect(objectContainer, 'g', 'lasso-container'); + var paths$$1 = container.selectAll('path[instance="'+instance+'"]'); + + // update + paths$$1 = paths$$1.data(allPoints); + + // remove excess + var pExit = paths$$1.exit().remove(); + // add needed paths + var pEnter = paths$$1.enter().append('path'); + + // merge + paths$$1 = paths$$1.merge(pEnter) + .transition().duration(transitionDuration) + .ease(easeFunc); + applyPathAttributes(paths$$1); + + } + + function applyPathAttributes(path) { + path + .attr("class", utils.str.hypenate(namespace, "lasso-path")) + .style('opacity', opacity) + .attr('fill', color$$1) + .attr("d", line) + .attr('instance', instance) + .style("stroke-dasharray", dashArray) + .attr("stroke", stroke) + .attr("stroke-width", strokeWidth) + .style('animation', 'lassoDash '+animationRate+' linear') + .style("animation-iteration-count", "infinite"); + } - /** - * Duration of all transitions of this element - * @param {number} [transitionDuration=1000] - * @memberof legend# - * @property - */ - transitionDuration = 1000, - /** - * Easing function for transitions - * @param {d3.ease} [easeFunc=d3.easeExp] - * @memberof legend# - * @property - */ - easeFunc = d3.easeExp; + function drag(event) { + /* + effectively create a mouse down and move event (which normally is inteperated + as 'drag' by the browser) by dynamically adding / removing this event on + mouse down / mouse up. + */ + if (eventCatcher != undefined) {eventCatcher.dispatch(utils.str.hypenate(namespace,"drag"));} + // d3.dispatch(utils.str.hypenate(namespace,"drag")) - legend.categories = function(_) { return arguments.length ? (categories=_, legend) : categories }; + if (event.which != 1) {return} // ensures left mouse button set + d3.event = event; + var pt = d3.mouse(objectContainer.node()); + var pt = d3.mouse(svg.node()); - /** - * Gets or sets the selection in which items are manipulated - * @param {d3.selection} [_=none] - * @returns {legend | d3.selection} - * @memberof legend - * @property - * by default selection = selection - */ - legend.selection = function(_) { return arguments.length ? (selection = _, legend) : selection; }; - /** - * Gets or sets the data - * (see {@link legend#data}) - * @param {number} [_=none] - * @returns {legend | object} - * @memberof legend - * @property - */ - legend.data = function(_) { return arguments.length ? (data = _, legend) : data; }; - /** - * Gets or sets the orient of the bars - * (see {@link legend#orient}) - * @param {number} [_=none] - * @returns {legend | object} - * @memberof legend - * @property - */ - legend.orient = function(_) { return arguments.length ? (orient = _, legend) : orient; }; - /** - * Gets or sets the amount of horizontal space in which items are manipulated - * (see {@link legend#spaceX}) - * @param {number} [_=none] should be a number > 0 - * @returns {legend | number} - * @memberof legend - * @property - * by default spaceX = undefined - */ - legend.spaceX = function(_) { return arguments.length ? (spaceX = _, legend) : spaceX; }; - /** - * Gets or sets the amount of vertical space in which items are manipulated - * (see {@link legend#spaceY}) - * @param {number} [_=none] should be a number > 0 - * @returns {legend | number} - * @memberof legend - * @property - * by default spaceY = undefined - */ - legend.spaceY = function(_) { return arguments.length ? (spaceY = _, legend) : spaceY; }; + if (xScale != undefined) {pt[0] = xScale.invert(pt[0]);} + if (yScale != undefined) {pt[1] = yScale.invert(pt[1]);} + pt[0] = pt[0] - chartOffset[0] - objectsOffset[0]; + pt[1] = pt[1] - chartOffset[1] - objectsOffset[1]; - /** - * Gets / sets whether or not legend is allowed to go beyond specified dimensions - * (see {@link legend#spaceX}) - * @param {boolean} [_=none] - * @returns {legend | boolean} - * @memberof legend - * @property - * by default overflowQ = false - */ - legend.overflowQ = function(_) { return arguments.length ? (overflowQ = _, legend) : overflowQ; }; - /** - * Gets / sets the grouping of the bars - * (see {@link legend#grouping}) - * @param {Array[]} [_=none] - * @returns {legend | Array[]} - * @memberof legend - * @property - * by default grouping = undefined - */ - legend.grouping = function(_) { return arguments.length ? (grouping = _, legend) : grouping; }; - /** - * Gets / sets the valueExtractor - * (see {@link legend#valueExtractor}) - * @param {function} [_=none] - * @returns {legend | function} - * @memberof legend - * @property - * by default valueExtractor = function(key, index) { return data[key] }, - */ - legend.valueExtractor = function(_) { return arguments.length ? (valueExtractor = _, legend) : valueExtractor; }; - /** - * Gets / sets the sortingFunction - * (see {@link bar#sortingFunction}) - * @param {function} [_=none] - * @returns {bar | function} - * @memberof bar - * @property - * by default sortingFunction = function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}, - */ - legend.sortingFunction = function(_) { return arguments.length ? (sortingFunction = _, legend) : sortingFunction; }; - /** - * Gets / sets objectSpacer - * (see {@link legend#objectSpacer}) - * @param {number} [_=none] - * @returns {legend | number} - * @memberof legend - * @property - * by default objectSpacer = 0.05 - */ - legend.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, legend) : objectSpacer; }; - /** - * Gets / sets the minObjectSize - * (see {@link legend#minObjectSize}) - * @param {number} [_=none] - * @returns {legend | number} - * @memberof legend - * @property - * by default minObjectSize = 50 - */ - legend.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, legend) : minObjectSize; }; - /** - * Gets / sets the maxObjectSize - * (see {@link legend#maxObjectSize}) - * @param {number} [_=none] - * @returns {legend | number} - * @memberof legend - * @property - * by default maxObjectSize = 100 - */ - legend.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, legend) : maxObjectSize; }; + /* if we have a point already, test if it passes a minimum distance to prevent overwhelming with too many tick functions */ + if (currentPoints.length) { + var lastPt = currentPoints[currentPoints.length - 1]; + var a = [pt[0], pt[1]], b = [lastPt[0], lastPt[1]]; - /** - * Gets / sets the barStrokeWidth - * (see {@link legend#barStrokeWidth}) - * @param {number} [_=none] - * @returns {legend | number} - * @memberof legend - * @property - * by default barStrokeWidth = 2 - */ - legend.bubbleStrokeWidth = function(_) { return arguments.length ? (bubbleStrokeWidth = _, legend) : bubbleStrokeWidth; }; - /** - * Gets / sets the colorFunction - * (see {@link legend#colorFunction}) - * @param {number} [_=none] - * @returns {legend | number} - * @memberof legend - * @property - * by default colorFunction = colorFunction() - */ - legend.colorFunction = function(_) { return arguments.length ? (colorFunction$$1 = _, legend) : colorFunction$$1; }; + if (xScale) {b[0] = xScale(b[0]); a[0] = xScale(a[0]);} + if (yScale) {b[1] = yScale(b[1]); a[1] = yScale(a[1]);} - /** - * Gets / sets the backgroundFill - * (see {@link legend#backgroundFill}) - * @param {string} [_=none] - * @returns {legend | string} - * @memberof legend - * @property - * by default backgroundFill = 'transparent' - */ - legend.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, legend) : backgroundFill; }; - /** - * Gets / sets the namespace - * (see {@link legend#namespace}) - * @param {string} [_=none] - * @returns {legend | string} - * @memberof legend - * @property - * by default namespace = 'd3sm-legend' - */ - legend.namespace = function(_) { return arguments.length ? (namespace = _, legend) : namespace; }; - /** - * Gets / sets the objectClass - * (see {@link legend#objectClass}) - * @param {string} [_=none] - * @returns {legend | string} - * @memberof legend - * @property - * by default objectClass = 'tick-group' - */ - legend.objectClass = function(_) { return arguments.length ? (objectClass = _, legend) : objectClass; }; - /** - * Gets / sets the transitionDuration - * (see {@link legend#transitionDuration}) - * @param {number} [_=none] - * @returns {legend | number} - * @memberof legend - * @property - * by default transitionDuration = 1000 - */ - legend.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, legend) : transitionDuration; }; - /** - * Gets / sets the easeFunc - * (see {@link legend#easeFunc}) - * @param {d3.ease} [_=none] - * @returns {legend | d3.ease} - * @memberof legend - * @property - * by default easeFunc = d3.easeExp - */ - legend.easeFunc = function(_) { return arguments.length ? (easeFunc = _, legend) : easeFunc; }; + var dist = utils.math.euclideanDistance(b, a); + if (dist > tickDistance) { tick(pt); } + } + else { tick(pt); } + } - function legend() { - var horizontalQ = (orient == 'horizontal') ? true : false; - var verticalQ = !horizontalQ; - // background cliping rectangle - var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; - var container = utils$1.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); + function tick (pt) { + /* + If a point is provided update data and objects. + Otherwise just call on data we already have. + Why like this?: + 1. currentPoints is current points to allow disjunct lassos, currentPoints is only pushed to + allPoints after mouseup. + 2. to allow render of objects in the lasso class / updating the data list + just by toggling the button + */ - colorFunction$$1.dataExtent([0, categories.length - 1]) - .colorBy('categories') - .categoryExtractor(function(k, v, i){return v}); + if (pt != undefined) { + currentPoints.push(pt); + path.attr("d", line); + if (currentPoints.length < 3) {return} // need at least 3 points to detect anything. + detect(allPoints.concat([currentPoints])); + } else { + detect(allPoints); + } + } - var r = Math.min(spaceX, spaceY) / 2; - var numberOfObjects = categories.length; - // if grouping is undefined sort barKeys by sortingFunction - var ordered = (grouping == undefined) ? categories.sort(sortingFunction) : grouping; - // ordered might be nested depending on grouping - var catKeys = utils$1.arr.flatten(ordered); + function detect(lassos) { + if (lassos == undefined) {lassos = allPoints;} + objectContainer.selectAll(objectClass).each(function(d, i){ + var current = d3.select(this), - var space = horizontalQ ? spaceX : spaceY; - // calculate object size - var objectSize = utils$1.math.calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ); - // calculate spacer size if needed - var spacerSize = utils$1.math.calculateWidthOfSpacer(catKeys, space, objectSize, numberOfObjects, objectSpacer, overflowQ); - // make the nested groups - var spacerFunction = groupingSpacer() - .horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects) - .objectClass(objectClass).objectSize(objectSize).spacerSize(spacerSize) - .transitionDuration(transitionDuration).easeFunc(easeFunc) - .namespace(namespace); + box = current.absolutePosition(), + // box = current.relativePositionTo(objectContainer.node()), - spacerFunction(container, ordered, 0); - var r = Math.min(objectSize, spaceX, spaceY) / 2 - bubbleStrokeWidth; + boxPts = [ + [ + box.left - chartOffset[0] - objectsOffset[0], + box.top - chartOffset[1] - objectsOffset[1] + ], + [ + box.right - chartOffset[0] - objectsOffset[0], + box.top - chartOffset[1] - objectsOffset[1] + ], + [ + box.left - chartOffset[0] - objectsOffset[0], + box.bottom - chartOffset[1] - objectsOffset[1] + ], + [ + box.right - chartOffset[0] - objectsOffset[0], + box.bottom - chartOffset[1] - objectsOffset[1] + ] + ]; - container.selectAll('g:not(.to-remove).'+objectClass).each(function(cat, i) { - var t = d3.select(this); - var c = utils$1.sel.safeSelect(t, 'circle'); - var fillColor = colorFunction$$1(undefined, cat, i, 'fill'), // prevent duplicate computation - strokeColor = colorFunction$$1(undefined, cat, i, 'stroke'); + if (xScale != undefined) { + boxPts[0][0] = xScale.invert(boxPts[0][0]); + boxPts[1][0] = xScale.invert(boxPts[1][0]); + boxPts[2][0] = xScale.invert(boxPts[2][0]); + boxPts[3][0] = xScale.invert(boxPts[3][0]); + } + if (yScale != undefined) { + boxPts[0][1] = yScale.invert(boxPts[0][1]); + boxPts[1][1] = yScale.invert(boxPts[1][1]); + boxPts[2][1] = yScale.invert(boxPts[2][1]); + boxPts[3][1] = yScale.invert(boxPts[3][1]); + } - var cx = horizontalQ - ? r+bubbleStrokeWidth - : (spaceX - r*2) / 2 + r; - var cy = verticalQ - ? r+bubbleStrokeWidth - : (spaceX - r*2) / 2 + r; - c.attr("r", r) - .attr('cx', cx) - .attr('cy', cy) - .attr('fill', fillColor) - .attr('stroke', strokeColor) - .attr('stroke-width', bubbleStrokeWidth); + /* + flag needed as we have to test multiple lasso segments, and if the point + is not in one segment, it does not mean it is not in any + */ + var inAnyLassoQ = false; + for (var i = 0; i < lassos.length; i++) { + var lassoPoints = lassos[i]; + // .map(function(pt){ + // var x, y = pt + // if (xScale!=undefined) {x = xScale(x)} + // if (yScale!=undefined) {y = yScale(y)} + // return [x, y] + // }) + var boxInLassoQ = boxPts.every(coord => d3.polygonContains(lassoPoints, coord)); - var text = utils$1.sel.safeSelect(t, 'text'); - text.text(cat) - .attr('text-anchor', 'middle') - .attr("transform", function(d, i){ - var - x = cx, - y = cy + text.node().getBoundingClientRect().height / 4, - t = 'translate('+x+','+y+')'; - return t - }); + if (boxInLassoQ) { inAnyLassoQ = true; } // only update flag in the positive case. + } + current.classed('in-lasso', inAnyLassoQ); + current.classed('in-lasso-'+instance, inAnyLassoQ); }); - + updateObjects(); + return objectContainer.selectAll('.in-lasso-'+instance) } - return legend -} -function numericLegend( selection ) { - var - min=0, - max=1, - spaceX, - spaceY, - colorFunction$$1 = colorFunction(), - namespace='d3sm-linear-vertical-gradient', - fontSize = 12, - backgroundFill = 'transparent', - textColor = 'black', - roundTo = 2; + function updateObjects() { + objectContainer.selectAll(objectClass).each(function(d, i) { + var t = d3.select(this); + applyObjectAttributes(t, t.classed('in-lasso')); + }); + } + function applyObjectAttributes(obj, setQ) { + var + preLassoFill = obj.attr('_pre_lasso_fill'), + preLassoStroke = obj.attr('_pre_lasso_stroke'), + preLassoStrokeWidth = obj.attr('_pre_lasso_stroke-width'); - legend.min = function(_) { return arguments.length ? (min=_, legend) : min }; - legend.max = function(_) { return arguments.length ? (max=_, legend) : max }; - legend.spaceX = function(_) { return arguments.length ? (spaceX=_, legend) : spaceX }; - legend.spaceY = function(_) { return arguments.length ? (spaceY=_, legend) : spaceY }; - legend.namespace = function(_) { return arguments.length ? (namespace=_, legend) : namespace }; - legend.fontSize = function(_) { return arguments.length ? (fontSize=_, legend) : fontSize }; - legend.backgroundFill = function(_) { return arguments.length ? (backgroundFill=_, legend) : backgroundFill }; - legend.colorFunction = function(_) { return arguments.length ? (colorFunction$$1=_, legend) : colorFunction$$1 }; - legend.textColor = function(_) { return arguments.length ? (textColor=_, legend) : textColor }; - legend.roundTo = function(_) { return arguments.length ? (roundTo=_, legend) : roundTo }; + if (setQ) { + obj.classed("in-lasso", true); + obj.classed('in-lasso-'+instance, true); + if (preLassoFill == undefined) { obj.attr('_pre_lasso_fill', obj.attr('fill')); } + if (preLassoStroke == undefined) { obj.attr('_pre_lasso_stroke', obj.attr('stroke')); } + if (preLassoStrokeWidth == undefined) { obj.attr('_pre_lasso_stroke-width', obj.attr('stroke-width')); } - function legend() { - // background cliping rectangle - var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; - var container = utils$1.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); + obj + //BUG: when .raise() + .attr('fill', lassoedFill) + .attr('stroke', lassoedStroke) + .attr('stoke-width', lassoedStrokeWidth); - var defs = utils$1.sel.safeSelect(selection, 'defs'); - var linearGradient = utils$1.sel.safeSelect(defs, 'linearGradient') - .attr("x1", "0%") - .attr("y1", "100%") - .attr("x2", "0%") - .attr("y2", "0%") - .attr('id', utils$1.str.hypenate(namespace,'numerical-legend-gradient')); + } else { + obj.classed("in-lasso", false); + obj.classed('in-lasso-'+instance, false); + if (preLassoFill != undefined) { obj.attr('fill', preLassoFill); } + if (preLassoStroke != undefined) { obj.attr('stroke', preLassoStroke); } + if (preLassoStrokeWidth != undefined) { obj.attr('stroke-width', preLassoStrokeWidth); } + } + } + function keyFrames() { + var style = + d3.select("html").select('style.'+utils.str.hypenate(namespace,"lasso-dash")); + if (style.empty()) { + d3.select("html").append('style') + .classed(utils.str.hypenate(namespace,"lasso-dash"), true) + .html("@keyframes lassoDash {to { stroke-dashoffset: 1000;}}"); + } - colorFunction$$1.dataExtent([min, max]) - .colorBy('value') - .valueExtractor(function(k, v, i){return v}); + } + return lasso +} +function lassoWidget( selection ) { + var + namespace = 'd3sm-lasso', + selection = selection, + svg, + chartContainer, + objectContainer, + objectClass, + lassoLine = d3.line() + .x(function(d, i){return d[0]}) + .y(function(d, i){return d[1]}) + .curve(d3.curveLinearClosed), + colorFunction$$1 = colorFunction(), + maxNumberOfGroups, + dataExtractor, + xScale, + yScale; - linearGradient.selectAll('stop') - .data( colorFunction$$1.colors() ) - .enter() - .append('stop') - .attr("offset", function(d, i){ return i / (colorFunction$$1.colors().length - 1) }) - .attr('stop-color', function(d) {return d}); + function onError(msg, style){ + console.log(msg, style); + } + // setup the container + setupDataselectContainer(); + + lassoWidget.objectClass = function(_){return arguments.length ? (objectClass='.'+_.replace('.',''), lassoWidget) : objectClass}; + lassoWidget.svg = function(_){return arguments.length ? (svg=_, lassoWidget) : svg}; + lassoWidget.submit = function(_){return arguments.length ? (submit=_, lassoWidget) : submit}; + lassoWidget.maxNumberOfGroups = function(_){return arguments.length ? (maxNumberOfGroups=_, lassoWidget) : maxNumberOfGroups}; + lassoWidget.onError = function(_){return arguments.length ? (onError=_, lassoWidget) : onError}; + lassoWidget.objectContainer = function(_){return arguments.length ? (objectContainer=_, lassoWidget) : objectContainer}; + lassoWidget.chartContainer = function(_){return arguments.length ? (chartContainer=_, lassoWidget) : chartContainer}; + lassoWidget.dataExtractor = function(_){return arguments.length ? (dataExtractor=_, lassoWidget) : dataExtractor}; + lassoWidget.xScale = function(_){return arguments.length ? (xScale=_, lassoWidget) : xScale}; + lassoWidget.yScale = function(_){return arguments.length ? (yScale=_, lassoWidget) : yScale}; + + function styles() { + var s = d3.select("html").select("style."+namespace+'lasso-widget'); + if (s.empty()) { + d3.select("html").append("style") + .classed(namespace+'lasso-widget', true) + .html( + "."+utils.str.hypenate(namespace, "data-table") + "{\ + height:100px;\ + overflow:auto;\ + }" + ); + } + } + function lassoWidget() { + styles(); + } - var rect = utils$1.sel.safeSelect(container, 'rect', 'legend') - .attr('transform', 'translate(0,'+fontSize+')') - .style("fill", "url(#"+utils$1.str.hypenate(namespace,'numerical-legend-gradient')+")") - .attr('x', 0) - .attr('y', 0) - .attr('width', spaceX) - .attr('height', spaceY - fontSize*2) - .on('mousemove', function(d, i){legendMousemove(d, i, rect, d3.select(this));}) - .on('mouseout', function(d, i){ d3.select("#"+utils$1.str.hypenate(namespace,'legend-tooltip')).remove(); }); + function submit(data) { + console.log(data); + } - var minText = utils$1.sel.safeSelect(container, 'text', 'min') - .text(utils$1.math.round(min, 2)) - .attr('text-anchor', 'middle') - .attr("font-size", fontSize+'px') - .attr('transform', function(d, i){ - var - x = spaceX / 2, - y = spaceY - fontSize / 4, - t = 'translate('+x+','+y+')'; - return t - }); + function plusTab(tabList) { + var tab = utils.sel.safeSelect(tabList, 'li', utils.str.hypenate(namespace, 'plus-tab')) + .classed('ml-auto', true) + .classed('nav-item', true) + .on('click', makeNewGroup), - var maxText = utils$1.sel.safeSelect(container, 'text', 'max') - .text(utils$1.math.round(max, 2)) - .attr('text-anchor', 'middle') - .attr("font-size", fontSize+'px') - .attr('transform', function(d, i){ - var - x = spaceX / 2, - y = fontSize, - t = 'translate('+x+','+y+')'; - return t - }); + anchor = utils.sel.safeSelect(tab, 'a', 'nav-link'), + icon = utils.sel.safeSelect(anchor, 'i', 'fa') + .classed('fa-plus fa-2x text-success', true); + } + + function sendTab(tabList) { + var tab = utils.sel.safeSelect(tabList, 'li', utils.str.hypenate(namespace, 'send-tab')) + .classed('ml-auto', true) + .classed('nav-item', true) + .on('click', clickSend) + , + anchor = utils.sel.safeSelect(tab, 'a', 'nav-link'), + icon = utils.sel.safeSelect(anchor, 'i', 'fa') + .classed('fa-paper-plane-o fa-2x text-primary', true); + } + function clickSend() { + var data = gatherDataLists(); + submit(data); + } + function closeTab(tabList) { + var tab = utils.sel.safeSelect(tabList, 'li', utils.str.hypenate(namespace, 'close-tab')) + .classed('ml-auto', true) + .classed('nav-item', true) + .on('click', function(){ + var m = d3.mouse(d3.select("html").node()); + selection.select('div.card').style("position", 'relative') + .transition().duration(1000) + .ease(d3.easeBack) + .style('left', window.outerWidth +'px') + .remove(); + }) + , + anchor = utils.sel.safeSelect(tab, 'a', 'nav-link'), + icon = utils.sel.safeSelect(anchor, 'i', 'fa') + .classed('fa-window-close-o fa-2x text-danger', true); } - function legendMousemove(d, i, rect, t) { - var s = d3.scaleLinear() - .domain([0, rect.attr('height')]) - .range([max, min]); - var m = d3.mouse(rect.node()); - var v = utils$1.math.round(s(m[1]),roundTo); - var strokeColor = colorFunction$$1(undefined, v, undefined, 'stroke'); - var fillColor = colorFunction$$1(undefined, v, undefined, 'fill'); - var div = utils$1.sel.safeSelect(d3.select('body'), 'div', utils$1.str.hypenate(namespace,'legend-tooltip')) - .attr('id', utils$1.str.hypenate(namespace,'legend-tooltip')) - .style('position', 'absolute') - .style('left', (d3.event.pageX+15)+'px') - .style('top', (d3.event.pageY+15)+'px') - .style('background-color', fillColor) - .style('border-color', strokeColor) + function setupDataselectContainer() { + var + card = utils.sel.safeSelect(selection, 'div', 'card'), + cardHeader = utils.sel.safeSelect(card, 'div', 'card-header'), - .style('min-width', (fontSize * (String(max).split('.')[0].length+3))+'px') - .style('min-height', (fontSize * (String(max).split('.')[0].length+3))+'px') - .style('border-radius', '50%') - .style('border-radius', '5000px') + tabList = utils.sel.safeSelect(cardHeader, 'ul', 'nav') + .classed('nav-tabs card-header-tabs', true) + .classed(utils.str.hypenate(namespace, 'tab-list'), true) + .attr('role', 'tablist'), - .style('display', 'flex') - .style('justify-content', 'center') - .style('text-align', 'middle') - .style('padding', 2+"px") + cardBody = utils.sel.safeSelect(card, 'div', 'card-body'), + tabContent = utils.sel.safeSelect(cardBody, 'div', 'tab-content') + .classed(utils.str.hypenate(namespace, 'tab-content'), true), - .style('border-style', 'solid') - .style('border-width', 2); + // leftTabs = utils.sel.safeSelect(tabList, 'div', 'nav mr-auto left-aligned-tabs') + rightTabs = utils.sel.safeSelect(tabList, 'div', 'right-aligned-tabs').classed('nav ml-auto ', true); - var text = utils$1.sel.safeSelect(div, 'div') - .text(v) - .style('color', textColor) - .style('align-self', 'center'); + plusTab(rightTabs); + sendTab(rightTabs); + closeTab(rightTabs); + var + defaultTab = utils.sel.safeSelect(tabContent, 'div', 'tab-pane') + .classed(utils.str.hypenate(namespace,'default-tab'), true) + .classed("active", true) + .classed('text-left', true), + + p = utils.sel.safeSelect(defaultTab, 'div') + .html( + "Click to add a new group.
"+ + "Click to submit for re-analysis.
"+ + "Click to close the dataselect widget." + ); + // .text('Click the green plus to add a new group.') } - return legend -} -let legends = { - categorical: categoricLegend, numeric: numericLegend -}; + function makeRemoveButton(paneBtnList) { + var + btn = utils.sel.safeSelect(paneBtnList, 'button', 'remove-btn') + .classed('btn btn-danger', true) + .on('click', removeBtnClickToRemove), + i = utils.sel.safeSelect(btn, 'i', 'fa').classed('fa-trash-o', true), + s = utils.sel.safeSelect(btn, 'span').text('Remove group'); + return btn + } + + function removeBtnClickToRemove(d, i) { + var + tab = selection.select('#'+utils.str.hypenate(namespace,'tab',d)), + pane = selection.select('#'+utils.str.hypenate(namespace,'tab','pane',d)); -// import * as d3 from "d3"; -/******************************************************************************* -** ** -** ** -** D3 EXTENSIONS ** -** ** -** ** -*******************************************************************************/ -/** -* Recursively ascends parents of selection until it finds an svg tag -* @function d3.selection.thisSVG -* @augments d3.selection -* @returns {Element} which is the svg tag, not the d3 selection of that tag -*/ -d3.selection.prototype.thisSVG = function() { return utils$1.sel.getContainingSVG(this.node()); }; + tab.remove(); + pane.remove(); + showRemainingPanes(); + updateTabAndPaneNumbers(); + } -/** -* Helper for getting absolute position of the mouse -* @function d3.mouse.absolute -* @augments d3.mouse -* @returns {number[]} [x, y] as they relate to `html` not to local scope. -*/ -d3.mouse.absolute = function() { - var html = d3.select('html').node(); - var [x, y] = this(html); - return [x, y] -}; + function togglePaneById(id) { + var panes = selection.select('.'+utils.str.hypenate(namespace, 'tab-content')); + panes.selectAll('div.tab-pane') + .each(function(d, i){ + d3.select(this).classed('active show', d3.select(this).attr('id') == id); + }); + } -/** -* Gets position of the selection in relation to the containing svg -* @see{@link getContainingSVG} -* @function d3.selection.absolutePosition -* @augments d3.selection -* @returns {Object} with structure similar to getBoundingClientRect, e.g. -* top, left, bottom, right, height, width -*/ -d3.selection.prototype.absolutePosition = function() { - var element = this.node(); - var elementPosition = element.getBoundingClientRect(); - var containerSVG = getContainingSVG(element); - var svgPosition = containerSVG.getBoundingClientRect(); + function insertTab(tabs, n) { + + // var li = tabs.insert("li", ':nth-child('+(n)+')') + var li = tabs.insert("li", '.right-aligned-tabs') + .classed('nav-item', true) + .classed('alert', true) + .classed('alert-secondary', true) + .classed('alert-dismissable', true) + .classed('fade', true) + .classed('show', true) + // .classed('mr-auto', true) + .attr("role", 'alert') + .attr("id", utils.str.hypenate(namespace,'tab',n)) + .classed(utils.str.hypenate(namespace,'tab'), true); + + var a = utils.sel.safeSelect(li, 'a') + .attr('data-toggle', 'tab') + .text('Group ' + n) + .attr('href', '#'+utils.str.hypenate(namespace,'tab','pane',n)) + .on('dblclick', function(d, i){ + var t = d3.select(this); + t.attr('contenteditable', true); + d3.select(t.node().parentNode) + .classed('alert-secondary', false) + .classed('alert-warning', true); - return { - top: elementPosition.top - svgPosition.top, - left: elementPosition.left - svgPosition.left, - bottom: elementPosition.bottom - svgPosition.top, - right: elementPosition.right - svgPosition.left, - height: elementPosition.height, - width: elementPosition.width - }; + // d3.select("html").on("click", dispatchBlur) + // + // function dispatchBlur(d, i) { + // if (d3.event.target != d3.select(this)) { + // d3.select(this).dispatch("blur") + // } + // } -}; + }) + .on('blur', function(d, i){ -d3.selection.prototype.relativePositionTo = function(container) { - var element = this.node(); - var elementPosition = element.getBoundingClientRect(); - var containerSVG = container; - var svgPosition = containerSVG.getBoundingClientRect(); + var t = d3.select(this); + t.attr('contenteditable', false); + d3.select(t.node().parentNode) + .classed('alert-secondary', true) + .classed('alert-warning', false); - return { - top: elementPosition.top - svgPosition.top, - left: elementPosition.left - svgPosition.left, - bottom: elementPosition.bottom - svgPosition.top, - right: elementPosition.right - svgPosition.left, - height: elementPosition.height, - width: elementPosition.width - }; + }) + .on('input', function(d, i){ + var t = d3.select(this); + var txt = t.text(); + d3.select(t.attr('href')).select("p.lead").text(txt); + }); -}; -// function getTranslation(selection){ -// var transform = selection.attr('transform') -// var [junk, xy] =transform.split('translate(') -// var [x, y] = xy.split(',') -// y, junk = y.split(')') -// return [parseFloat(x), parseFloat(y)] -// } + return li + } -function lasso( selection ) { - var - svg, // svg that is target of events - objectContainer, // container which houses objects we are selecting (allows for transform to be applied to lasso) - objectClass, // class of object we are selecting - namespace="d3sm-lasso", - chartContainer, - chartOffset, - objectsOffset, - eventCatcher, + function makeTabPane(panes, n) { + var pane = panes.append('div') + .datum(n) + .attr('role', 'tabpanel') + .classed('tab-pane', true) + .classed('text-left', true) + .classed(utils.str.hypenate(namespace,'tab','pane'), true) + .attr('id', utils.str.hypenate(namespace,'tab','pane',n)); + return pane + } - xScale, // optional scale for the lasso currentPoints - yScale, // optional scale for the lasso currentPoints + function populatePane(pane, n) { + var + lead = utils.sel.safeSelect(pane, 'p', 'lead').text('Group '+n), - activeQ = false, // whether or not lasso is active + tableContainer = utils.sel.safeSelect(pane, 'div', 'table-responsive') + .attr("class", utils.str.hypenate(namespace, "data-table")), - currentPoints=[], // mouse points for current lasso - allPoints=[], // list of lists for all points of lassos + table = utils.sel.safeSelect(tableContainer, "table", "table") + .classed("table-sm", true).classed("table-hover", true), - line = d3.line() - .x(function(d, i){ - var x; - if (xScale != undefined) { x = xScale(d[0]); } - else {x = d[0];} - return x //- chartOffset[0]// - objectsOffset[0] - }) - .y(function(d, i){ - var y; - if (yScale != undefined) { y= yScale(d[1]); } - else {y = d[1];} - return y// - chartOffset[1]// - objectsOffset[1] - }) - .curve(d3.curveLinearClosed), + caption = utils.sel.safeSelect(table, "caption", "caption").html("List of selected"), - instance=0, // an indentifier for which instance this lasso is under the current svg + btns = utils.sel.safeSelect(pane, 'div', 'text-right'), - tickDistance = 10, + cN = n % colorFunction$$1.colors().length; + if (n % 2 != 0) { cN = (colorFunction$$1.colors().length - 1) - cN; } - // styles for lasso path - color$$1 = '#17a2b8', - animationRate = '10s', - opacity=0.3, - dashArray = '5, 10', - stroke = 'black', - strokeWidth=2, - // styles for lassoed objects - lassoedFill = "white", - lassoedStroke = 'black', - lassoedStrokeWidth = 3, - transitionDuration = 1000, - easeFunc = d3.easeExp; - var path; - lasso.svg = function(_) { return arguments.length ? (svg = _, lasso) : svg; }; - lasso.chartContainer = function(_) { return arguments.length ? (chartContainer = _, lasso) : chartContainer; }; - lasso.objectContainer = function(_) { return arguments.length ? (objectContainer = _, lasso) : objectContainer; }; - lasso.objectClass = function(_) { return arguments.length ? (objectClass = _, lasso) : objectClass; }; - lasso.namespace = function(_) { return arguments.length ? (namespace = _, lasso) : namespace; }; - lasso.xScale = function(_) { return arguments.length ? (xScale = _, lasso) : xScale; }; - lasso.yScale = function(_) { return arguments.length ? (yScale = _, lasso) : yScale; }; - lasso.activeQ = function(_) { return arguments.length ? (activeQ = _, lasso) : activeQ; }; - lasso.currentPoints = function(_) { return arguments.length ? (currentPoints = _, lasso) : currentPoints; }; - lasso.allPoints = function(_) { return arguments.length ? (allPoints = _, lasso) : allPoints; }; - lasso.instance = function(_) { return arguments.length ? (instance = _, lasso) : instance; }; - lasso.tickDistance = function(_) { return arguments.length ? (tickDistance = _, lasso) : tickDistance; }; - lasso.color = function(_) { return arguments.length ? (color$$1 = _, lasso) : color$$1; }; - lasso.animationRate = function(_) { return arguments.length ? (animationRate = _, lasso) : animationRate; }; - lasso.opacity = function(_) { return arguments.length ? (opacity = _, lasso) : opacity; }; - lasso.dashArray = function(_) { return arguments.length ? (dashArray = _, lasso) : dashArray; }; - lasso.stroke = function(_) { return arguments.length ? (stroke = _, lasso) : stroke; }; - lasso.lassoedFill = function(_) { return arguments.length ? (lassoedFill = _, lasso) : lassoedFill; }; - lasso.lassoedStroke = function(_) { return arguments.length ? (lassoedStroke = _, lasso) : lassoedStroke; }; - lasso.lassoedStrokeWidth = function(_) { return arguments.length ? (lassoedStrokeWidth = _, lasso) : lassoedStrokeWidth; }; - lasso.eventCatcher = function(_) { return arguments.length ? (eventCatcher = _, lasso) : eventCatcher; }; + var lassoBtn = makeLassoButton(btns); + var currentLasso = makeLassoFunction(n, colorFunction$$1.colors()[cN]); + var clearBtn = makeClearButton(btns); - lasso.drag = drag; - lasso.draw = draw; - lasso.tick = tick; - lasso.detect = detect; - lasso.toggle = toggle; - lasso.remove = remove; - lasso.render = render; - lasso.keyFrames = keyFrames; - lasso.updateObjects = updateObjects; - lasso.applyPathAttributes = applyPathAttributes; - lasso.applyObjectAttributes = applyObjectAttributes; + bindButtons(lassoBtn, clearBtn, currentLasso, table); - keyFrames(); + var removeBtn = makeRemoveButton(btns); + + lead.on("mouseover", function(){ + currentLasso.draw(); + }); + lead.on("mouseout", function(){ + currentLasso.remove(); + }); - function lasso() { - // add a dash animation if needed - if (activeQ) { transitionDraw(); } } - function toggle(state) { - // use optional param to set state, otherwise toggle state - activeQ = (state!=undefined) ? state : !activeQ; - chartOffset = utils$1.sel.getTranslation(chartContainer); - objectsOffset = utils$1.sel.getTranslation(objectContainer); + function makeLassoButton(btns) { + var btn = utils.sel.safeSelect(btns, 'button', 'lasso-btn') + .classed('btn btn-info', true) + .classed(namespace, true); + // .datum([lasso]) + var i = utils.sel.safeSelect(btn, 'i', 'fa').classed('fa-hand-pointer-o', true); + var s = utils.sel.safeSelect(btn, 'span').text('Lasso select'); + return btn - if (activeQ) { - svg.node().addEventListener('mousedown', render, true); - } else { - svg.node().removeEventListener('mousedown', render, true); - remove(); - } + } + function makeClearButton(btns) { + var btn = utils.sel.safeSelect(btns, 'button', 'clear-btn') + .classed('btn btn-light', true) + .classed(namespace, true); + var i = utils.sel.safeSelect(btn, 'i', 'fa').classed('fa-eraser', true); + var s = utils.sel.safeSelect(btn, 'span').text('Clear selection'); + return btn } - function draw() { - chartOffset = utils$1.sel.getTranslation(chartContainer); - objectsOffset = utils$1.sel.getTranslation(objectContainer); + function makeLassoFunction(instance, color$$1) { + var currentLasso = lasso() + .namespace(namespace) + .svg(svg) + .objectClass(objectClass) + .chartContainer(chartContainer) + .objectContainer(objectContainer) + .instance(instance) + .color(color$$1) + .yScale(yScale) + .xScale(xScale); + + return currentLasso + } - var container = utils$1.sel.safeSelect(objectContainer, 'g', 'lasso-container'); - var paths$$1 = container.selectAll('path[instance="'+instance+'"]'); + function bindButtons(lassoBtn, clearBtn, currentLasso, table) { + currentLasso.eventCatcher(lassoBtn); + lassoBtn.node().addEventListener("click", lassoBtnToggle); + + lassoBtn.datum([currentLasso]) + .on(utils.str.hypenate(namespace,'render'), function(){ + d3.select(this).datum()[0].draw(); + }) + .on(utils.str.hypenate(namespace,"drag"), function(las){ + var nodes = chartContainer.selectAll(".in-lasso-"+las[0].instance()).nodes(); + var tableData = nodes.map(function(d, i){ + var extracted = dataExtractor(d3.select(d).datum()); + extracted["__node"] = d; + return extracted + }); - // update - paths$$1 = paths$$1.data(allPoints); - // remove excess - var pExit = paths$$1.exit().remove(); - // add needed paths - var pEnter = paths$$1.enter().append('path'); + makeTable(table, tableData, currentLasso); + }); - // merge - paths$$1 = paths$$1.merge(pEnter); + clearBtn.on('click', function(){ + currentLasso.allPoints([]); + currentLasso.currentPoints([]); + lassoBtn.dispatch(utils.str.hypenate(namespace,"drag")); - // apply - applyPathAttributes(paths$$1); - } + }); - function remove() { - var container = utils$1.sel.safeSelect(objectContainer, 'g', 'lasso-container'); - var paths$$1 = container.selectAll('path[instance="'+instance+'"]').remove(); - container.remove(); - objectContainer.selectAll(objectClass).classed("in-lasso", false); - updateObjects(); } - function render( event ) { - // nothing can interefer with drawing the lasso - event.preventDefault(); event.stopPropagation(); + function lassoBtnToggle() { + var that = d3.select(this); + var las = that.datum()[0]; - var container = utils$1.sel.safeSelect(objectContainer, 'g', 'lasso-container'); + las.toggle(); + var activeQ = las.activeQ(); - /* - each time the user presses down, while the state is active, the lasso - the lasso should make a seperate segment. - */ - currentPoints = []; + if (las.activeQ()) { + that.classed('btn-info', !activeQ); + that.classed('btn-warning', activeQ); + that.select("span").text("Lasso select (active)"); + selection.selectAll("."+namespace+".lasso-btn").dispatch(utils.str.hypenate(namespace,'render')); + d3.select("html").node().addEventListener('mousedown', monitorLassoButtonState); + } else { + that.classed('btn-info', !activeQ); + that.classed('btn-warning', activeQ); + that.select("span").text("Lasso select"); + d3.select("html").node().removeEventListener('mousedown', monitorLassoButtonState); + } - svg.node().addEventListener('mousemove', drag); - svg.node().addEventListener('mouseup', function(event) { - svg.node().removeEventListener('mousemove', drag); - allPoints.push(currentPoints); - // BUG: somehow this is pushing currentPoints n times where n is the nth lasso path for the current instance - // NOTE: allPoints = utils.arr.unique(allPoints) is a temporary and inefficient fix - allPoints = utils$1.arr.unique(allPoints); - }); + function monitorLassoButtonState(event) { + /* + activeLasso stops event stopPropagation, so this event will not register in + that case. Thus we only need to ensure that the usre is clicking on the lasso button + otherwise, de-spawn lasso. + */ + if ( + event.target != that.node() && + event.target != that.select("span").node() && + event.target != that.select("i").node() + ) { + // event.preventDefault() + // event.stopPropagation() + las.toggle(false); + that.classed('btn-info', true); + that.classed('btn-warning', false); + that.select("span").text("Lasso select"); + // console.log(that, that.node()) + // that.dispatch("focusout") + } + } - path = container.append('path').data([currentPoints]); - applyPathAttributes(path); } - function transitionDraw() { - var container = utils$1.sel.safeSelect(objectContainer, 'g', 'lasso-container'); - var paths$$1 = container.selectAll('path[instance="'+instance+'"]'); - // update - paths$$1 = paths$$1.data(allPoints); + function updateTableHeaderColumns(headR, tableData) { + var headerKeys = d3.keys(tableData[0]).filter(k=>k!="__node"); + if (headerKeys.length > 0) { + // headerKeys = ["remove"].concat(headerKeys) + headerKeys.push("remove"); + } - // remove excess - var pExit = paths$$1.exit().remove(); - // add needed paths - var pEnter = paths$$1.enter().append('path'); + var headerCols = headR.selectAll("th"); - // merge - paths$$1 = paths$$1.merge(pEnter) - .transition().duration(transitionDuration) - .ease(easeFunc); - applyPathAttributes(paths$$1); + headerCols = headerCols.data(headerKeys); + headerCols.exit().remove(); + headerCols = headerCols.merge(headerCols.enter().append("th").attr("scope","col")) + .text(function(d, i){return d}); + return headerKeys } - function applyPathAttributes(path) { - path - .attr("class", utils$1.str.hypenate(namespace, "lasso-path")) - .style('opacity', opacity) - .attr('fill', color$$1) - .attr("d", line) - .attr('instance', instance) - .style("stroke-dasharray", dashArray) - .attr("stroke", stroke) - .attr("stroke-width", strokeWidth) - .style('animation', 'lassoDash '+animationRate+' linear') - .style("animation-iteration-count", "infinite"); + function updateTableRows(body, tableData) { + var bodyRows = body.selectAll("tr"); + + bodyRows = bodyRows.data(tableData); + bodyRows.exit().remove(); + bodyRows = bodyRows.merge(bodyRows.enter().append("tr")); + + return bodyRows } - function drag(event) { - /* - effectively create a mouse down and move event (which normally is inteperated - as 'drag' by the browser) by dynamically adding / removing this event on - mouse down / mouse up. - */ + function updateTableRowColumns(cols, headerKeys, rowData, tableData, lasso$$1, table) { + cols = cols.data(headerKeys); + cols.exit().remove(); + cols = cols.merge(cols.enter().append("td")); + + cols.html(function(d, i){ + if (d != "remove") { return rowData[d] } + return "" + }).classed("text-left", function(d, i){ + if (d != "remove") { return false } + return true + }); + // .attr("scope",function(d, i){ + // if (d != "remove") { return false } + // return true + // }) + + cols.select("i.fa-close").on("click", function(d, i){ + var node = rowData["__node"], + n = d3.select(node); + n.classed("in-lasso", false); + n.classed("in-lasso-"+lasso$$1.instance(), false); + + tableData.map(function(dd, j){ + if (dd["__node"] == node) { + tableData.splice(j, 1); + makeTable(table, tableData, lasso$$1); + } + }); + + }); + } + + function makeTable(table, tableData, lasso$$1) { + var + head = utils.sel.safeSelect(table, "thead"), + headR = utils.sel.safeSelect(head, "tr"), + body = utils.sel.safeSelect(table, "tbody"), + + headerKeys = updateTableHeaderColumns(headR, tableData), - if (eventCatcher != undefined) {eventCatcher.dispatch(utils$1.str.hypenate(namespace,"drag"));} - // d3.dispatch(utils.str.hypenate(namespace,"drag")) + bodyRows = updateTableRows(body, tableData); - if (event.which != 1) {return} // ensures left mouse button set - d3.event = event; - var pt = d3.mouse(objectContainer.node()); - var pt = d3.mouse(svg.node()); + bodyRows.each(function(rowData, i){ + var t = d3.select(this); + var cols = t.selectAll("td"); - if (xScale != undefined) {pt[0] = xScale.invert(pt[0]);} - if (yScale != undefined) {pt[1] = yScale.invert(pt[1]);} - pt[0] = pt[0] - chartOffset[0] - objectsOffset[0]; - pt[1] = pt[1] - chartOffset[1] - objectsOffset[1]; + updateTableRowColumns(cols, headerKeys, rowData, tableData, lasso$$1, table); - /* if we have a point already, test if it passes a minimum distance to prevent overwhelming with too many tick functions */ - if (currentPoints.length) { - var lastPt = currentPoints[currentPoints.length - 1]; - var a = [pt[0], pt[1]], b = [lastPt[0], lastPt[1]]; + t.on("mouseover", function(d, i) { + lasso$$1.applyObjectAttributes(d3.select(d["__node"]),true); + }) + .on("mouseout", function(d, i) { + lasso$$1.applyObjectAttributes(d3.select(d["__node"]),false); + }); + }); - if (xScale) {b[0] = xScale(b[0]); a[0] = xScale(a[0]);} - if (yScale) {b[1] = yScale(b[1]); a[1] = yScale(a[1]);} - var dist = utils$1.math.euclideanDistance(b, a); - if (dist > tickDistance) { tick(pt); } - } - else { tick(pt); } } - function tick (pt) { - /* - If a point is provided update data and objects. - Otherwise just call on data we already have. - Why like this?: - 1. currentPoints is current points to allow disjunct lassos, currentPoints is only pushed to - allPoints after mouseup. - 2. to allow render of objects in the lasso class / updating the data list - just by toggling the button - */ - if (pt != undefined) { - currentPoints.push(pt); - path.attr("d", line); - if (currentPoints.length < 3) {return} // need at least 3 points to detect anything. - detect(allPoints.concat([currentPoints])); - } else { - detect(allPoints); - } - } - function detect(lassos) { - if (lassos == undefined) {lassos = allPoints;} - objectContainer.selectAll(objectClass).each(function(d, i){ - var current = d3.select(this), - box = current.absolutePosition(), - // box = current.relativePositionTo(objectContainer.node()), + function makeNewGroup(d, i) { - boxPts = [ - [ - box.left - chartOffset[0] - objectsOffset[0], - box.top - chartOffset[1] - objectsOffset[1] - ], - [ - box.right - chartOffset[0] - objectsOffset[0], - box.top - chartOffset[1] - objectsOffset[1] - ], - [ - box.left - chartOffset[0] - objectsOffset[0], - box.bottom - chartOffset[1] - objectsOffset[1] - ], - [ - box.right - chartOffset[0] - objectsOffset[0], - box.bottom - chartOffset[1] - objectsOffset[1] - ] - ]; + var tabs = selection.select('.'+utils.str.hypenate(namespace, 'tab-list')), + panes = selection.select('.'+utils.str.hypenate(namespace, 'tab-content')), + n = newGroupNumber(); - if (xScale != undefined) { - boxPts[0][0] = xScale.invert(boxPts[0][0]); - boxPts[1][0] = xScale.invert(boxPts[1][0]); - boxPts[2][0] = xScale.invert(boxPts[2][0]); - boxPts[3][0] = xScale.invert(boxPts[3][0]); - } - if (yScale != undefined) { - boxPts[0][1] = yScale.invert(boxPts[0][1]); - boxPts[1][1] = yScale.invert(boxPts[1][1]); - boxPts[2][1] = yScale.invert(boxPts[2][1]); - boxPts[3][1] = yScale.invert(boxPts[3][1]); - } + if (tabs.selectAll('.'+utils.str.hypenate(namespace,'tab')).size() == maxNumberOfGroups) { + onError('only '+maxNumberOfGroups+' allowed.', 'warning'); + return + } + var + tab = insertTab(tabs, n), + pane = makeTabPane(panes, n); - /* - flag needed as we have to test multiple lasso segments, and if the point - is not in one segment, it does not mean it is not in any - */ - var inAnyLassoQ = false; - for (var i = 0; i < lassos.length; i++) { - var lassoPoints = lassos[i]; - // .map(function(pt){ - // var x, y = pt - // if (xScale!=undefined) {x = xScale(x)} - // if (yScale!=undefined) {y = yScale(y)} - // return [x, y] - // }) - var boxInLassoQ = boxPts.every(coord => d3.polygonContains(lassoPoints, coord)); + populatePane(pane, n); + togglePaneById(pane.attr('id')); - if (boxInLassoQ) { inAnyLassoQ = true; } // only update flag in the positive case. - } + } - current.classed('in-lasso', inAnyLassoQ); - current.classed('in-lasso-'+instance, inAnyLassoQ); + function newGroupNumber() { + var tabs = selection.select('.'+utils.str.hypenate(namespace, 'tab-list'));//, + var + n = tabs.selectAll('li').size() - 3, // minus 1 for plus tab + s = 'Group ' + n, + gs = []; + tabs.each(function(d, i){ return gs.push(d3.select(this).select("a").text()) }); + while (gs.includes(s)) { n+=1; s = 'Group ' + n; } + return n + } + + function updateTabAndPaneNumbers() { + var + tabs = selection.select('.'+utils.str.hypenate(namespace, 'tab-list')), + panes = selection.select('.'+utils.str.hypenate(namespace, 'tab-content')); + + tabs = tabs.selectAll('.'+utils.str.hypenate(namespace,'tab')); + panes = panes.selectAll('.'+utils.str.hypenate(namespace,'tab', 'pane')); + + tabs.each(function(d, i){ + d3.select(this).datum(i) + .attr("id", utils.str.hypenate(namespace,'tab',i)) + .select('a') + .attr('href', '#'+utils.str.hypenate(namespace,'tab','pane',i)) + .text(function(dd, ii){ + var curText = d3.select(this).text(); + if (curText.split(' ')[0] == 'Group') { + return 'Group ' + i + } + return curText + }); + }); + + panes.each(function(d, i){ + d3.select(this).datum(i) + .attr('id', utils.str.hypenate(namespace,'tab','pane',i)); + utils.sel.safeSelect(d3.select(this), 'p', 'lead') + .text(function(dd, ii){ + var curText = d3.select(this).text(); + if (curText.split(' ')[0] == 'Group') { + return 'Group ' + i + } + return curText + }); + utils.sel.safeSelect(d3.select(this), 'button', 'remove-btn') + .on('click', removeBtnClickToRemove); }); - updateObjects(); - return objectContainer.selectAll('.in-lasso-'+instance) } + function gatherDataLists() { + var tabs = selection.select('.'+utils.str.hypenate(namespace, 'tab-list')); + var panes = selection.select('.'+utils.str.hypenate(namespace, 'tab-content')); + var tables = panes.selectAll('.'+utils.str.hypenate(namespace, "data-table")); + var data = {}; + var textGroups = tabs.selectAll('li.'+utils.str.hypenate(namespace,'tab') + ' > a') + .nodes().map(function(d, i){return d3.select(d).text()}); - function updateObjects() { - objectContainer.selectAll(objectClass).each(function(d, i) { - var t = d3.select(this); - applyObjectAttributes(t, t.classed('in-lasso')); + textGroups.map(function(e, i){ + data[e] = []; }); - } - function applyObjectAttributes(obj, setQ) { - var - preLassoFill = obj.attr('_pre_lasso_fill'), - preLassoStroke = obj.attr('_pre_lasso_stroke'), - preLassoStrokeWidth = obj.attr('_pre_lasso_stroke-width'); - if (setQ) { - obj.classed("in-lasso", true); - obj.classed('in-lasso-'+instance, true); - if (preLassoFill == undefined) { obj.attr('_pre_lasso_fill', obj.attr('fill')); } - if (preLassoStroke == undefined) { obj.attr('_pre_lasso_stroke', obj.attr('stroke')); } - if (preLassoStrokeWidth == undefined) { obj.attr('_pre_lasso_stroke-width', obj.attr('stroke-width')); } + tables.each(function(d, i){ + var table = d3.select(this).select("tbody"); + data[textGroups[i]] = table.selectAll('tr').data(); + }); - obj - //BUG: when .raise() - .attr('fill', lassoedFill) - .attr('stroke', lassoedStroke) - .attr('stoke-width', lassoedStrokeWidth); + return data + } - } else { - obj.classed("in-lasso", false); - obj.classed('in-lasso-'+instance, false); - if (preLassoFill != undefined) { obj.attr('fill', preLassoFill); } - if (preLassoStroke != undefined) { obj.attr('stroke', preLassoStroke); } - if (preLassoStrokeWidth != undefined) { obj.attr('stroke-width', preLassoStrokeWidth); } + function showRemainingPanes() { + var + tabs = selection.select('.'+utils.str.hypenate(namespace, 'tab-list')), + panes = selection.select('.'+utils.str.hypenate(namespace, 'tab-content')), + remainingTabs = tabs.selectAll('.'+utils.str.hypenate(namespace,'tab')); + + if (remainingTabs.size() == 0) { + panes.select('.'+utils.str.hypenate(namespace,'default-tab')) + .classed("active", true) + .classed('text-left', true); + } + else { + var lastTab = remainingTabs.nodes()[remainingTabs.size()-1], + lastPaneId = d3.select(lastTab).select('a').attr('href'); + panes.select(lastPaneId) + .classed("active", true) + .classed('text-left', true); } } - function keyFrames() { - var style = - d3.select("html").select('style.'+utils$1.str.hypenate(namespace,"lasso-dash")); - if (style.empty()) { - d3.select("html").append('style') - .classed(utils$1.str.hypenate(namespace,"lasso-dash"), true) - .html("@keyframes lassoDash {to { stroke-dashoffset: 1000;}}"); - } - } - return lasso + return lassoWidget } /******************************************************************************* @@ -8064,14 +9386,14 @@ function multiPlotZoom( chart ) { function setLocks() { - var chartObjSel = chartSel.select('.'+utils$1.str.hypenate(chart.namespace(),'object-container')); - var chartObjTrans = utils$1.sel.getTranslation(chartObjSel.attr('transform')); + var chartObjSel = chartSel.select('.'+utils.str.hypenate(chart.namespace(),'object-container')); + var chartObjTrans = utils.sel.getTranslation(chartObjSel.attr('transform')); var cos = chartObjSel.attr('transform', 'translate(0,0)'); xLock = chartSel.node().getBBox().width - chart.spaceX();// * .9 yLock = chartSel.node().getBBox().height - chart.spaceY();// * .9 cos.attr('transform', 'translate('+chartObjTrans[0]+','+chartObjTrans[1]+')'); - utils$1.con.log('plotZoom', 'setLocks', {xLock:xLock, yLock:yLock}); + utils.con.log('plotZoom', 'setLocks', {xLock:xLock, yLock:yLock}); } @@ -8131,20 +9453,20 @@ function multiPlotZoom( chart ) { - var chartObjSel = chartSel.select('.'+utils$1.str.hypenate(chart.namespace(),'object-container')); + var chartObjSel = chartSel.select('.'+utils.str.hypenate(chart.namespace(),'object-container')); var xComponentObjSel = xComponentsSel.map(function(d, i){ - return d.select('.'+utils$1.str.hypenate(xComponents[i].namespace(),'object-container')) + return d.select('.'+utils.str.hypenate(xComponents[i].namespace(),'object-container')) }); var yComponentObjSel = yComponentsSel.map(function(d, i){ - return d.select('.'+utils$1.str.hypenate(yComponents[i].namespace(),'object-container')) + return d.select('.'+utils.str.hypenate(yComponents[i].namespace(),'object-container')) }); - var chartObjTrans = utils$1.sel.getTranslation(chartObjSel.attr('transform')); + var chartObjTrans = utils.sel.getTranslation(chartObjSel.attr('transform')); var xComponentsObjTrans = xComponentObjSel.map(function(d, i){ - return utils$1.sel.getTranslation(d.attr('transform')) + return utils.sel.getTranslation(d.attr('transform')) }); var yComponentsObjTrans = yComponentObjSel.map(function(d, i){ - return utils$1.sel.getTranslation(d.attr('transform')) + return utils.sel.getTranslation(d.attr('transform')) }); var x = horizontalQ ? transform.applyX(chartObjTrans[0]) : 0; @@ -8169,9 +9491,9 @@ function multiPlotZoom( chart ) { zoom.reset = function() { - var chartObjSel = chartSel.select('.'+utils$1.str.hypenate(chart.namespace(),'object-container')); - var xAxisObjSel = xAxisSel.select('.'+utils$1.str.hypenate(xAxis.namespace(),'object-container')); - var yAxisObjSel = yAxisSel.select('.'+utils$1.str.hypenate(yAxis.namespace(),'object-container')); + var chartObjSel = chartSel.select('.'+utils.str.hypenate(chart.namespace(),'object-container')); + var xAxisObjSel = xAxisSel.select('.'+utils.str.hypenate(xAxis.namespace(),'object-container')); + var yAxisObjSel = yAxisSel.select('.'+utils.str.hypenate(yAxis.namespace(),'object-container')); chartObjSel.attr('transform', 'translate('+0+','+0+')'); xAxisObjSel.attr('transform', 'translate('+0+','+0+')'); yAxisObjSel.attr('transform', 'translate('+0+','+0+')'); @@ -8302,13 +9624,13 @@ function plotZoom( chart, xAxis, yAxis ) { function setLocks() { - var chartObjSel = chartSel.select('.'+utils$1.str.hypenate(chart.namespace(),'object-container')); - var chartObjTrans = utils$1.sel.getTranslation(chartObjSel.attr('transform')); + var chartObjSel = chartSel.select('.'+utils.str.hypenate(chart.namespace(),'object-container')); + var chartObjTrans = utils.sel.getTranslation(chartObjSel.attr('transform')); var cos = chartObjSel.attr('transform', 'translate(0,0)'); xLock = chartSel.node().getBBox().width - chart.spaceX() * .9; yLock = chartSel.node().getBBox().height - chart.spaceY() * .9; cos.attr('transform', 'translate('+chartObjTrans[0]+','+chartObjTrans[1]+')'); - utils$1.con.log('plotZoom', 'setLocks', {xLock:xLock, yLock:yLock}); + utils.con.log('plotZoom', 'setLocks', {xLock:xLock, yLock:yLock}); } @@ -8367,9 +9689,9 @@ function plotZoom( chart, xAxis, yAxis ) { - var chartObjSel = chartSel.select('.'+utils$1.str.hypenate(chart.namespace(),'object-container')); - var xAxisObjSel = xAxisSel.select('.'+utils$1.str.hypenate(xAxis.namespace(),'object-container')); - var yAxisObjSel = yAxisSel.select('.'+utils$1.str.hypenate(yAxis.namespace(),'object-container')); + var chartObjSel = chartSel.select('.'+utils.str.hypenate(chart.namespace(),'object-container')); + var xAxisObjSel = xAxisSel.select('.'+utils.str.hypenate(xAxis.namespace(),'object-container')); + var yAxisObjSel = yAxisSel.select('.'+utils.str.hypenate(yAxis.namespace(),'object-container')); // xLock = chartSel.node().getBBox().width - chart.spaceX() - chartSel.node().getBBox().x // yLock = chartSel.node().getBBox().height - chart.spaceY() @@ -8377,9 +9699,9 @@ function plotZoom( chart, xAxis, yAxis ) { // bhm.selection().node().getBBox().width - bhm.spaceX() - var chartObjTrans = utils$1.sel.getTranslation(chartObjSel.attr('transform')); - var xAxisObjTrans = utils$1.sel.getTranslation(xAxisObjSel.attr('transform')); - var yAxisObjTrans = utils$1.sel.getTranslation(yAxisObjSel.attr('transform')); + var chartObjTrans = utils.sel.getTranslation(chartObjSel.attr('transform')); + var xAxisObjTrans = utils.sel.getTranslation(xAxisObjSel.attr('transform')); + var yAxisObjTrans = utils.sel.getTranslation(yAxisObjSel.attr('transform')); var x = horizontalQ ? transform.applyX(chartObjTrans[0]) : 0; @@ -8401,9 +9723,9 @@ function plotZoom( chart, xAxis, yAxis ) { zoom.reset = function() { - var chartObjSel = chartSel.select('.'+utils$1.str.hypenate(chart.namespace(),'object-container')); - var xAxisObjSel = xAxisSel.select('.'+utils$1.str.hypenate(xAxis.namespace(),'object-container')); - var yAxisObjSel = yAxisSel.select('.'+utils$1.str.hypenate(yAxis.namespace(),'object-container')); + var chartObjSel = chartSel.select('.'+utils.str.hypenate(chart.namespace(),'object-container')); + var xAxisObjSel = xAxisSel.select('.'+utils.str.hypenate(xAxis.namespace(),'object-container')); + var yAxisObjSel = yAxisSel.select('.'+utils.str.hypenate(yAxis.namespace(),'object-container')); chartObjSel.attr('transform', 'translate('+0+','+0+')'); xAxisObjSel.attr('transform', 'translate('+0+','+0+')'); yAxisObjSel.attr('transform', 'translate('+0+','+0+')'); @@ -8412,11 +9734,297 @@ function plotZoom( chart, xAxis, yAxis ) { return zoom } +function selectFilter(selection) { + + var + data, + namespace = 'd3sm-select-filter', + selectionName = 'Select options:', + defaultValue = undefined; + + + + + var lastValue = undefined; + + selectFilter.data = function(_) { return arguments.length ? (data = _, selectFilter) : data}; + selectFilter.namespace = function(_) { return arguments.length ? (namespace = _, selectFilter) : namespace}; + selectFilter.selectionName = function(_) { return arguments.length ? (selectionName = _, selectFilter) : selectionName}; + selectFilter.defaultValue = function(_) { return arguments.length ? (defaultValue = _, selectFilter) : defaultValue}; + selectFilter.currentOption = currentOption; + + function selectFilter() { + var + container = utils.sel.safeSelect(selection, 'div', 'input-group').classed(utils.str.hypenate(namespace,'container'),true), + + selectPrepend = utils.sel.safeSelect(container, 'div', 'select-prepend').classed('input-group-prepend', true), + selectPrependSpan = utils.sel.safeSelect(selectPrepend, 'span', 'input-group-text').text(selectionName), + + select = utils.sel.safeSelect(container, 'select', 'custom-select').classed(utils.str.hypenate(namespace,'select'),true), + + selectAppend = utils.sel.safeSelect(container, 'div', 'select-append').classed('input-group-prepend', true), + selectAppendButton = utils.sel.safeSelect(selectAppend, 'a', 'filter-button').classed('btn btn-outline-secondary', true), + filterButtonIcon = utils.sel.safeSelect(selectAppendButton, 'i', 'fa fa-filter'), + + inputGroup = utils.sel.safeSelect(container, 'div', 'filter-input-group').classed('input-group',true).classed('d-none', true), + inputPrepend = utils.sel.safeSelect(inputGroup, 'div', 'input-group-prepend'), + inputPrependSpan = utils.sel.safeSelect(inputPrepend, 'span', 'input-group-text').classed('search-button', true), + inputPrependSpanIcon = utils.sel.safeSelect(inputPrependSpan,'i','fa fa-search'), + + input = utils.sel.safeSelect(inputGroup, 'input', 'form-control').attr('placeholder', 'all').attr('type', 'text'), + inputAppend = utils.sel.safeSelect(inputGroup, 'div', 'input-group-append'), + inputAppendButton = utils.sel.safeSelect(inputAppend, 'a', 'close-button').classed('btn btn-outline-secondary', true), + inputAppendButtonIcon = utils.sel.safeSelect(inputAppendButton, 'i', 'fa fa-close'); + + + var keys = d3.keys(data), + options = select.selectAll('option'); + + options = options.data(d3.keys(data)); + options = options.merge(options.enter().append('option')) + .attr('value', function(d, i){return d}) + .text(function(d, i){return d}); + + var + filterButton = selectAppendButton, + closeButton = inputAppendButton; + + filterButton.on('click', function(d, i){ + var currentStyle = inputGroup.classed('d-none'); + inputGroup.classed('d-none', !currentStyle); + }); + + closeButton.on('click', function(d, i){ + input.property('value', '').dispatch('input'); + }); + + input.on('input', function(d, i){ + var + val = input.property('value'), + reg = new RegExp(val, 'gi'), + use; + + if (val == '') {use = keys;} + else { + use = []; + d3.keys(data).map(function(option, j){ + var match = option.match(reg); + if (match == null || match.join('') == '') ; + else { use.push(option); } + }); + } + + options = select.selectAll('option'); + options = options.data(use); + options.exit().remove(); + options = options.merge(options.enter().append('option')) + .attr('value', function(d, i){return d}) + .text(function(d, i){return d}); + + var current = currentOption(); + if (lastValue != current) { + lastValue = current; + select.dispatch('change'); + } + }); + + + } + + function currentOption() { + var val = selection.select("select").property('value'); + return val == undefined || val == '' + ? defaultValue == undefined + ? d3.keys(data)[0] + : defaultValue + : val + } + + return selectFilter +} + +/******************************************************************************* +** ** +** ** +** DATATOGGLE ** +** ** +** ** +*******************************************************************************/ +/** + * Creates a datatoggle + * @constructor datatoggle + * @param {d3.selection} selection + * @namespace datatoggle + * @returns {function} datatoggle + */ +function datatoggle( selection ) { + var + /** + * Keys to make toggle-able options + * (see {@link datatoggle#keys}) + * @param {string[]} [keys=undefined] + * @memberof datatoggle# + * @property + */ + keys, + + /** + * What to do when a different key is clicked + * (see {@link datatoggle#updateFunction}) + * @param {function} [updateFunction=function(){}] + * @memberof datatoggle# + * @property + */ + updateFunction = function(){}, + /** + * Namespace for all items made by this instance of datatoggle + * @param {string} [namespace="d3sm-databar"] + * @memberof datatoggle# + * @property + */ + namespace='d3sm-databar', + /** + * Currently toggled key + * @param {string} [currentKey=undefined] + * @memberof datatoggle# + * @property + */ + currentKey, + + + + xAxisSelectQ = false, + xAxisOptions, + yAxisSelectQ = false, + yAxisOptions, + data; + toggle.xAxisSelectQ = function(_){return arguments.length ? (xAxisSelectQ = _, toggle) : xAxisSelectQ}; + toggle.yAxisSelectQ = function(_){return arguments.length ? (yAxisSelectQ = _, toggle) : yAxisSelectQ}; + toggle.xAxisOptions = function(_){return arguments.length ? (xAxisOptions = _, toggle) : xAxisOptions}; + toggle.yAxisOptions = function(_){return arguments.length ? (yAxisOptions = _, toggle) : yAxisOptions}; + toggle.data = function(_){return arguments.length ? (data = _, toggle) : data}; + toggle.keys = function(_){return arguments.length ? (keys = _, toggle) : keys}; + toggle.currentKey = function(_){return arguments.length ? (currentKey = _, toggle) : currentKey}; + + /** + * Gets / sets the updateFunction + * (see {@link datatoggle#updateFunction}) + * @function datatoggle.updateFunction + * @param {function} [_=none] + * @returns {datatoggle | function} + * @memberof datatoggle + * @property + * by default updateFunction = function(){} + */ + toggle.updateFunction = function(_){return arguments.length ? (updateFunction = _, toggle) : updateFunction}; + /** + * Gets / sets the namespace + * (see {@link datatoggle#namespace}) + * @function datatoggle.namespace + * @param {string} [_=none] + * @returns {datatoggle | string} + * @memberof datatoggle + * @property + * by default namespace = 'd3sm-databar' + */ + toggle.namespace = function(_){return arguments.length ? (namespace = _, toggle) : namespace}; + /** + * Gets / sets the currentKey + * (see {@link datatoggle#currentKey}) + * @function datatoggle.currentKey + * @param {string} [_=none] + * @returns {datatoggle | string} + * @memberof datatoggle + * @property + * by default currentKey = undefined + */ + toggle.currentKeys = + function () { + var vals = {}; + d3.keys(filterSelects).map(function(k, i){ + vals[k]= filterSelects[k].currentOption(); + }); + return vals + }; + + var filterSelects = {}; + function toggle() { + // selection options + + // selection.classed('d-flex flex-row', true) + // var filterButton = utils.sel.safeSelect(selection, 'a', 'slider-buttons') + // var filterI = utils.sel.safeSelect(filterButton, 'i', 'fa fa-sliders') + + /*BUG: unexpected behavior. + - when using bootstrap-eque way for collapse, clicking button submits to + the same page in applications (but not in demo), so using anchor () + - when using anchor, cause page to jump to that location + - when using show, the first open works, but the close does not. + */ + // filterButton.attr('class', 'btn btn-secondary') + // .attr('data-toggle', 'collapse') + // .attr('href', '#'+utils.str.hypenate(namespace, 'data-toggle')) + // .attr('target', '_blank') + // .html(filterButton.html()+' Filters') + // .on('click', function(d, i){ + // d3.event.preventDefault() + // d3.event.stopPropagation() + // var dt = d3.select("#"+utils.str.hypenate(namespace, 'data-toggle')) + // dt.classed('show', !dt.classed('show')) + // dt.classed('d-inline-flex', dt.classed('show')) + // + // filterButton.classed('btn-primary', dt.classed('show')) + // filterButton.classed('btn-secondary', !dt.classed('show')) + // }) + // .classed('d-inline-flex', true) + // .style("margin-right", '10px') + + + + // var datatoggleCollapse = utils.sel.safeSelect(selection, 'div', utils.str.hypenate(namespace,'collapse')) + // .attr('id', utils.str.hypenate(namespace, 'data-toggle')) + // .classed('collapse', true) + + // var flexRow = utils.sel.safeSelect(datatoggleCollapse, 'div', 'd-inline-flex flex-row flex-wrap') + var flexRow = utils.sel.safeSelect(selection, 'div', 'd-inline-flex flex-row flex-wrap'); + + var dataopts = flexRow.selectAll('div.'+utils.str.hypenate(namespace,'select-filter')); + // remove excess + dataopts.exit().remove(); + // bind data + dataopts = dataopts.data(d3.keys(data)); + //enter + var doEnter = dataopts.enter().append('div') + .attr('class', 'select-filter'); + + dataopts = dataopts.merge(doEnter).style('margin-right', "10px"); + + dataopts.each(function(d, i){ + var t = d3.select(this); + var sf = selectFilter(t) + .data(data[d]) + .namespace(utils.str.hypenate(namespace, d)) + .selectionName(d); + sf(); + filterSelects[d] = sf; + }); + + + selection.selectAll('select') + .on('change', function(){updateFunction();}); + // bind update function + // d3.selectAll + return toggle + } + + return toggle +} + let aux = { - lasso, plotZoom, multiPlotZoom + lasso, plotZoom, multiPlotZoom, datatoggle, selectFilter, lassoWidget }; -let d3sm$1 = { +let d3sm = { aux, axis, charts, @@ -8425,11 +10033,11 @@ let d3sm$1 = { groupingSpacer, legends, tooltip, - utils: utils$1, + utils, }; if (typeof window !== 'undefined') { - window.d3sm = d3sm$1; + window.d3sm = d3sm; } -export default d3sm$1; +export default d3sm; diff --git a/dist/d3sm.min.js b/dist/d3sm.min.js index 283cfaa..78dd43f 100644 --- a/dist/d3sm.min.js +++ b/dist/d3sm.min.js @@ -1 +1 @@ -!function(t){"use strict";function e(t,e,n){return n.indexOf(t)===e}function n(t,e){return void 0==e?t.every(function(t){return!0===t}):t.every(function(t){return e(t)})}function r(t){var e={};return t.map(function(t){o(Object.keys(e),t)?e[t]=1:e[t]+=1}),e}function o(t,e){return t.includes(e)}function a(t){return t[0]}function i(t){return t[t.length-1]}function l(t){return t.reduce((t,e)=>t+e,0)}function c(t){return t.filter(e)}function u(t,e){return t.filter(function(t,n){return o(e,n)})}function s(t){return n(t.map(function(t,e){return Array.isArray(t)}))}function h(t,e){return s(t)?e.map(function(e,n){return t.get(e)}):u(t,e)}function f(t,e,n){void 0==n&&(n=c(t.map(function(t,n){return element[e]}))).map(function(t,e){r[t]=[]});var r={};return t.map(function(t,n){r[t[e]].push(t)}),r}function d(t,e){if(!e)return!1;if(t.length!=e.length)return!1;for(var n=0,r=t.length;n=n.length?n.push(t.length):n[e]+=t.length,t.map(function(t,r){Array.isArray(t)&&g(t,e,n)}),n}function p(t,e){return e=void 0==e?0:e,t.map(function(t,n){Array.isArray(t)?e=p(t,e):e+=1}),e}function m(t,e){return e=void 0==e?[]:e,t.map(function(t,n){Array.isArray(t)?e=e.concat(m(t)):e.push(t)}),e}function v(t,e){for(var n=0;ntt>n),a=void 0==(a=d3.median(r))?n:a,i=void 0==(i=d3.min(r))?a:i,l=void 0==(l=d3.median(o))?n:l,c=void 0==(c=d3.max(o))?l:c,u="q0",s="q1",h="q2",f="q3",d="q4",g={};return void 0!=e&&5==e.length&&(u=e[0],s=e[1],h=e[2],f=e[3],d=e[4]),g[u]=i,g[s]=a,g[h]=n,g[f]=l,g[d]=c,g}function z(t,e,n,r,o,a){var i=t-(e-1)*(o=0==o||o>1?o:t*o),l=(i=i<0?0:i)/e;return a&&void 0!=n&&l=n.length?n.push(t.length-1):n[e]+=t.length-1,t.map(function(t,r){Array.isArray(t)&&A(t,e,n)}),n}function C(t,e){var n=E(function(){t()},e);window.addEventListener("resize",n)}function E(t,e,n){var r;return function(){var o=this,a=arguments,i=function(){r=null,n||t.apply(o,a)},l=n&&!r;clearTimeout(r),r=setTimeout(i,e),l&&t.apply(o,a)}}function L(t,e,n,r={top:.01,bottom:.01,left:.01,right:.01},o={w:.8,h:.6},a={y:.1,x:.1},i={x:0,margin:0,pos:"left"}){void 0==n&&(n={w:window.innerWidth,h:window.Height});var l={w:n.w*o.w,h:n.h*o.h},c={top:r.top*l.h,bottom:r.bottom*l.h,left:r.left*l.w,right:r.right*l.w},u={w:l.w-c.left-c.right,h:l.h-c.top-c.bottom},s={x:u.h*a.x,y:u.w*a.y},h={x:u.w-s.y-i.x-2*i.margin,y:u.h-s.x},f={x:i.margin+c.left+("left"==i.pos?0:h.x+s.y),y:c.top,w:i.x,h:h.y},d={x:s.y+c.left+("left"==i.pos?i.x+2*i.margin:0),y:c.top,w:s.y,h:h.y},g={x:s.y+c.left+("left"==i.pos?i.x+2*i.margin:0),y:c.top,w:h.x,h:h.y},p={x:s.y+c.left+("left"==i.pos?i.x+2*i.margin:0),y:c.top+h.y,w:h.x,h:s.x};n=d3sm.safeSelect(t,"svg",e).style("width",l.w+"px").style("height",l.h+"px");var a=d3sm.safeSelect(n,"g",d3sm.hypenate(e,"axes")),i=d3sm.safeSelect(n,"g",d3sm.hypenate(e,"legend")).attr("transform","translate("+f.x+","+f.y+")");return{svg:{selection:n,rect:l},plot:{selection:d3sm.safeSelect(n,"g",d3sm.hypenate(e,"plot")).attr("transform","translate("+g.x+","+g.y+")"),rect:g},xAxis:{selection:d3sm.safeSelect(a,"g",d3sm.hypenate(e,"x-axis")).attr("transform","translate("+p.x+","+p.y+")"),rect:p},yAxis:{selection:d3sm.safeSelect(a,"g",d3sm.hypenate(e,"y-axis")).attr("transform","translate("+d.x+","+d.y+")"),rect:d},legend:{selection:i,rect:f}}}function M(){return Array.prototype.slice.call(arguments).join("-")}function F(t,e,n,r,o,a){var i=t.node().getBoundingClientRect();for(t.text(e);Math.max(i.width,i.height)>o-r&&(e=String(e),e=e.slice(0,e.length-1),t.text(e+"..."),i=t.node().getBoundingClientRect(),0!=e.length););}function B(t,e,n){var r=e/n;return r g.'+l+"."+s);"function"==typeof f?v.each(function(t,e){f(d3.select(this))}):v.remove()}else{x+=n;var m=d.select("g."+s+'[level="'+g+'"] > g.'+l+"."+s);m.empty()&&(m=d.append("g").attr("class",l).classed(s,!0)),m.attr("parent-index",r);var v=d.selectAll("g."+s+'[level="'+(g+1)+'"]');"function"==typeof f?v.each(function(t,e){f(d3.select(this))}):v.remove()}x+=r==p.size()-1?0:y}),x}var e,n,r,o=!0,a=d3.scaleLinear(),i="category",l="d3sm-groupped-item",c=1e3,u=d3.easeSin,s="spacer",h=function(t){t.attr("transform",function(t,e){return"translate("+(o?window.outerWidth:0)+","+(o?0:window.outerWidth)+")"})},f=function(t){K("groupingSpacer","exiting with",{current:t,currentNode:t.node()}),t.selectAll("g").classed("to-remove",!0),t.transition().duration(.9*c).ease(u).attr("transform",function(t,e){return"translate("+(o?window.outerWidth:0)+","+(o?0:window.outerWidth)+")"}).remove()};return t.horizontalQ=function(e){return arguments.length?(o=e,t):o},t.scale=function(e){return arguments.length?(a=e,t):a},t.moveby=function(e){return arguments.length?(i=e,t):i},t.numberOfObjects=function(n){return arguments.length?(e=n,t):e},t.objectClass=function(e){return arguments.length?(l=e,t):l},t.objectSize=function(e){return arguments.length?(n=e,t):n},t.spacerSize=function(e){return arguments.length?(r=e,t):r},t.transitionDuration=function(e){return arguments.length?(c=e,t):c},t.easeFunc=function(e){return arguments.length?(u=e,t):u},t.namespace=function(e){return arguments.length?(s=e,t):s},t.enterFunction=function(e){return arguments.length?(h=e,t):h},t.exitFunction=function(e){return arguments.length?(f=e,t):f},t}function G(t){function e(){function e(t,e,n,r,o){return n&&r?o/2:0}function W(t,e,n,r,o){return n&&r?o/2:0}var D=!!gt.arr.hasQ(["top","bottom","horizontal"],d),R=!D,G={x:0,y:0,width:g,height:p};"left"==d&&(G.x-=g,y&&(G.width+=u),G.y-=V,G.height+=2*V),"bottom"==d&&(G.y=G.y,y&&(G.y-=u,G.height+=u),G.x-=V,G.width+=2*V),"top"==d&&(G.y-=p,y&&(G.height+=u),G.y-=V,G.height+=2*V),"right"==d&&(G.x=0,y&&(G.width+=u,G.x-=u),G.y-=V,G.height+=2*V);var Z=gt.sel.setupContainer(t,z,G,j);"top"==d&&(l=void 0==l?"start":l,c=void 0==c?-90:c),"bottom"==d&&(l=void 0==l?"end":l,c=void 0==c?-90:c),"left"==d&&(l=void 0==l?"end":l,c=void 0==c?0:c),"right"==d&&(l=void 0==l?"start":l,c=void 0==c?0:c);var I=v?void 0==o?a:o:void 0==o?void 0!=A?gt.math.tickRange(...d3.extent(i),A):i:o,J=gt.arr.flatten(I),U=J.length,$=D?g:p,tt=d3.extent(J);N&&tt.reverse();var et=N?[tt[0]+b,tt[1]-b]:[tt[0]-b,tt[1]+b];x.domain(et).range([D?0:p,D?g:0]),s=void 0==s?gt.math.calculateWidthOfObject($,U,w,S,k,m):s,h=void 0==h?gt.math.calculateWidthOfSpacer(J,$,s,U,k,m):h;var nt=gt.str.hypenate(z,v?O+"-categorical":O),rt=H().horizontalQ(D).scale(x).moveby(v?"category":"scale").numberOfObjects(U).objectClass(nt).objectSize(s).spacerSize(h).transitionDuration(X).easeFunc(P).namespace(z),ot=function(t){var e=x(t.datum()),n=2*x(tt[1]),r=ewindow.innerWidth&&o.style("left",d3.event.pageX-15-300+"px")}function r(t,e){var n=d3.select(this).style("fill","black");if(d3.select(n.node().parentNode).select("line."+gt.str.hypenate(z,"tick")).attr("stroke",L).attr("stroke-width",M),y){var r=d3.select(n.node().parentNode).select("line."+gt.str.hypenate(z,"guideline")),o=r.attr("minor");r.attr("stroke",function(t,e){return"true"==o?gt.color.modifyHexidecimalColorLuminance(Y,.8):Y}).attr("stroke-width",function(t,e){return"true"==o?.8*K:K})}d3.select("#"+gt.str.hypenate(z,"guideline-tooltip")).remove()}var o,a,i,l,c,u,s,h,f,d="bottom",g=0,p=0,m=!1,v=!1,y=!1,x=d3.scaleLinear(),b=.5,k=.05,w=15,S=50,j="transparent",z="d3sm-axis",O="tick-group",A=5,C="black",E=3,L="black",M=2,F=10,B=10,Q=10,q=14,W=8,V=20,D=void 0,T=function(t,e){},R=function(t,e){return String(t).replace("-"," ").replace("_"," ")},Y="#333333",K=2,X=1e3,P=d3.easeExp,_=2,N=!1;return e.label=function(t){return arguments.length?(f=t,e):f},e.tickTickLabelSpacer=function(t){return arguments.length?(B=t,e):B},e.tickLabelMargin=function(t){return arguments.length?(Q=t,e):Q},e.selection=function(n){return arguments.length?(t=n,e):t},e.orient=function(t){return arguments.length?(d=t,e):d},e.spaceX=function(t){return arguments.length?(g=t,e):g},e.spaceY=function(t){return arguments.length?(p=t,e):p},e.overflowQ=function(t){return arguments.length?(m=t,e):m},e.categoricalQ=function(t){return arguments.length?(v=t,e):v},e.guideLinesQ=function(t){return arguments.length?(y=t,e):y},e.grouping=function(t){return arguments.length?(o=t,e):o},e.scale=function(t){return arguments.length?(x=t,e):x},e.domainPadding=function(t){return arguments.length?(b=t,e):b},e.objectSpacer=function(t){return arguments.length?(k=t,e):k},e.minObjectSize=function(t){return arguments.length?(w=t,e):w},e.maxObjectSize=function(t){return arguments.length?(S=t,e):S},e.namespace=function(t){return arguments.length?(z=t,e):z},e.backgroundFill=function(t){return arguments.length?(j=t,e):j},e.objectClass=function(t){return arguments.length?(O=t,e):O},e.tickLabels=function(t){return arguments.length?(a=t,e):a},e.tickValues=function(t){return arguments.length?(i=t,e):i},e.numberOfTicks=function(t){return arguments.length?(A=t,e):A},e.lineStroke=function(t){return arguments.length?(C=t,e):C},e.lineStrokeWidth=function(t){return arguments.length?(E=t,e):E},e.tickStroke=function(t){return arguments.length?(L=t,e):L},e.tickStrokeWidth=function(t){return arguments.length?(M=t,e):M},e.tickLength=function(t){return arguments.length?(F=t,e):F},e.tickLabelFontSize=function(t){return arguments.length?(q=t,e):q},e.tickLabelMinFontSize=function(t){return arguments.length?(W=t,e):W},e.tickLabelMaxFontSize=function(t){return arguments.length?(V=t,e):V},e.tickLabelTextAnchor=function(t){return arguments.length?(l=t,e):l},e.tickLabelRotation=function(t){return arguments.length?(c=t,e):c},e.tickLabelFunc=function(t){return arguments.length?(D=t,e):D},e.tickLabelOnClick=function(t){return arguments.length?(T=t,e):T},e.guidelineSpace=function(t){return arguments.length?(u=t,e):u},e.guideLineStroke=function(t){return arguments.length?(Y=t,e):Y},e.guideLineStrokeWidth=function(t){return arguments.length?(K=t,e):K},e.transitionDuration=function(t){return arguments.length?(X=t,e):X},e.easeFunc=function(t){return arguments.length?(P=t,e):P},e.objectSize=function(t){return arguments.length?(s=t,e):s},e.spacerSize=function(t){return arguments.length?(h=t,e):h},e.roundTo=function(t){return arguments.length?(_=t,e):_},e.reverseScaleQ=function(t){return arguments.length?(N=t,e):N},e.tickLabelOnHoverFunc=function(t){return arguments.length?(R=t,e):R},e}function Z(t){function e(){t.on("mouseover",n),t.on("mousemove",n),t.on("mouseout",function(){d3.selectAll(".d3sm-tooltip").remove()})}function n(t,e){R("d3sm-tooltip");var n=i[t],[l,c]=d3.mouse(d3.select("html").node());K("tooltip","mousemove detected",{key:t,index:e,x:l,y:c}),K("tooltip","current data",n);var u=W(d3.select("html"),"tooltip","d3sm-tooltip").classed("card",!0).style("max-width","300px").style("background-color","#212529").style("color","white"),s=W(u,"div","card-body"),h=(W(s,"h5","card-title").text(void 0==a?t:"function"==typeof a?a(t,n,e):a).style("color","cyan"),W(W(s,"table","table").classed("table-dark",!0),"tbody"));(h=(h=h.selectAll("tr")).data(void 0==r?d3.keys(n):r)).exit().remove();var f=h.enter().append("tr").style("max-width","300px");f.append("td").attr("class",function(t,e){return"tooltip-key"}),f.append("td").attr("class",function(t,e,n){return"tooltip-value"}).attr("tooltip-row-index",function(t,e){return e}),R("tooltip-rows"),h.selectAll(".tooltip-key").text(function(t,e){return t}),h.selectAll("tr .tooltip-value").text(function(t,e){K("tooltip","trying to set value",{rowKey:t,rowIndex:e});var e=d3.select(this).attr("tooltip-row-index"),r=n[t];return void 0!=o&&"function"==typeof(r=o[e])&&(r=r(n,t)),"number"==typeof r?b(r,5):r}),Y(),Y(),l+=15;var d=u.node().getBoundingClientRect();l+d.width>window.innerWidth-window.scrollX&&(l=d3.event.pageX-d.width-15),c+d.height>window.innerHeight-window.scrollY&&(c=d3.event.pageY-d.height-15),"relative"==u.style("position")?u.style("position","absolute").style("left",l+"px").style("top",c+"px"):u.style("left",l+"px").style("top",c+"px"),u.attr("z-index",1e4)}var r,o,a,i,t;return e.keys=function(t){return arguments.length?(r=t,e):r},e.values=function(t){return arguments.length?(o=t,e):o},e.header=function(t){return arguments.length?(a=t,e):a},e.data=function(t){return arguments.length?(i=t,e):i},e.selection=function(n){return arguments.length?(t=n,e):t},e}function I(){function t(t,r,o,u,d){var p,m="fill"==u?l:i;if(e(),"index"==c)p=void 0!=u?a(g(f(o)),m):g(f(o));else if("value"==c){y=s(t,r,o);p=void 0!=u?a(g(f(y)),m):g(f(y))}else if("category"==c){var v=h(t,r,o),y=n.indexOf(v);p=void 0!=u?a(g(f(y)),m):g(f(y))}else p=void 0!=u?a(g(f(o)),m):g(f(o));return p}function e(){d.domain([0,r.length]),"category"==c&&void 0!=n?d.range([0,n.length]):d.range(u);var t=Array(r.length).fill(0).map(function(t,e){return d(e)});f.domain(t)}var n,r=["#2c7bb6","#00a6ca","#00ccbc","#90eb9d","#ffff8c","#f9d057","#f29e2e","#e76818","#d7191c"],o=d3.interpolateRgb,a=y,i=0,l=.4,c="index",u=[0,r.length-1],s=function(t,e,n){return e},h=function(t,e,n){return e.category},f=d3.scaleLinear().interpolate(o).domain(u).range(r),d=d3.scaleLinear(),g=function(t){return"#"+t.match(/\d+/g).map(function(t,e){return(+t<16?"0":"")+(+t).toString(16)}).join("")};return t.colors=function(e){return arguments.length?(r=e,f.range(r),t):r},t.interpolation=function(e){return arguments.length?(o=e,f.interpolate(o).range(r),t):o},t.dataExtent=function(e){return arguments.length?(u=e,f.domain(u).interpolate(f.interpolate()),t):u},t.scale=function(e){return arguments.length?(e=e.domain(f.domain()).interpolate(f.interpolate()).range(f.range()),f=e,t):f},t.modifyOpacity=function(e){return arguments.length?(a=e,t):a},t.strokeOpacity=function(e){return arguments.length?(i=e,t):i},t.fillOpacity=function(e){return arguments.length?(l=e,t):l},t.colorBy=function(e){return arguments.length?(c=e,t):c},t.valueExtractor=function(e){return arguments.length?(s=e,t):s},t.categoryExtractor=function(e){return arguments.length?(h=e,t):h},t.categories=function(e){return arguments.length?(n=e,t):n},t}function J(t){function e(){var e={x:0,y:0,width:r,height:o},C=gt.sel.setupContainer(t,S,e,w);a=d3.keys(n),i=a.map(h),l=a.map(g),c=a.map(v);a.length;var E=[Math.min(...i)-s,Math.max(...i)+s],L=[Math.min(...l)-d,Math.max(...l)+d],M=[Math.min(...c)-m,Math.max(...c)+m];u.domain(E).range([0,r]),f.domain(L).range([o,0]),p.domain(M).range([y,x]);var F=C.selectAll("."+j),B=(F=F.data(a)).enter().append("circle").attr("class",j).attr("cx",0).attr("cy",o).attr("r",0),Q=F.exit();(F=F.merge(B)).each(function(t,e){var r=d3.select(this),o=n[t],a=i[e],s=l[e],h=c[e],d=k(t,o,e,"fill"),g=k(t,o,e,"stroke");r.transition().duration(z).ease(O).attr("cx",u(a)).attr("cy",f(s)).attr("r",p(h)).attr("fill",d).attr("stroke",g).attr("stroke-width",b),r.on("mouseover",function(t,e){F.style("opacity",.2),r.style("opacity",1),r.transition().duration(z/2).ease(O).attr("stroke-width",2*b).attr("r",1.5*p(h))}),r.node().addEventListener("mouseout",function(){C.selectAll("."+j).style("opacity",1),r.transition().duration(z/2).ease(O).attr("stroke-width",b).attr("r",p(h))})}),Q.transition().duration(z).ease(O).attr("cx",0).attr("cy",o).attr("r",0).remove(),A.selection(F).data(n),A()}var n,r,o,a,i,l,c,u=d3.scaleLinear(),s=.5,h=function(t,e){return n[t].x},f=d3.scaleLinear(),d=.5,g=function(t,e){return n[t].y},p=d3.scaleLinear(),m=.5,v=function(t,e){return 2},y=2,x=10,b=2,k=I(),w="transparent",S="d3sm-scatter",j="scatter-point",z=1e3,O=d3.easeExp,A=Z();return e.selection=function(n){return arguments.length?(t=n,e):t},e.data=function(t){return arguments.length?(n=t,e):n},e.spaceX=function(t){return arguments.length?(r=t,e):r},e.spaceY=function(t){return arguments.length?(o=t,e):o},e.scaleX=function(t){return arguments.length?(u=t,e):u},e.domainPaddingX=function(t){return arguments.length?(s=t,e):s},e.valueExtractorX=function(t){return arguments.length?(h=t,e):h},e.scaleY=function(t){return arguments.length?(f=t,e):f},e.domainPaddingY=function(t){return arguments.length?(d=t,e):d},e.valueExtractorY=function(t){return arguments.length?(g=t,e):g},e.scaleR=function(t){return arguments.length?(p=t,e):p},e.domainPaddingR=function(t){return arguments.length?(m=t,e):m},e.valueExtractorR=function(t){return arguments.length?(v=t,e):v},e.minRadius=function(t){return arguments.length?(y=t,e):y},e.maxRadius=function(t){return arguments.length?(x=t,e):x},e.pointStrokeWidth=function(t){return arguments.length?(b=t,e):b},e.colorFunction=function(t){return arguments.length?(k=t,e):k},e.backgroundFill=function(t){return arguments.length?(w=t,e):w},e.namespace=function(t){return arguments.length?(S=t,e):S},e.objectClass=function(t){return arguments.length?(j=t,e):j},e.transitionDuration=function(t){return arguments.length?(z=t,e):z},e.easeFunc=function(t){return arguments.length?(O=t,e):O},e.pointKeys=function(t){return arguments.length?(a=t,e):a},e.valuesX=function(t){return arguments.length?(i=t,e):i},e.valuesY=function(t){return arguments.length?(l=t,e):l},e.valuesR=function(t){return arguments.length?(c=t,e):c},e.tooltip=function(t){return arguments.length?(A=t,e):A},e}function U(t){function e(){var e="horizontal"==s||"bottom"==s||"top"==s,C=!e,E={x:0,y:0,width:r,height:o},L=gt.sel.setupContainer(t,w,E,k);i=d3.keys(n),l=i.map(f);var M=void 0==a?i.sort(d):a,F=(i=gt.arr.flatten(M)).length,B=[Math.min(...l)-p,Math.max(...l)+p];g.domain(B).range(e?[0,o]:"right"==s?[0,r]:[r,0]);var Q=e?r:o;c=void 0==c?gt.math.calculateWidthOfObject(Q,F,v,y,m,h):c,u=void 0==u?gt.math.calculateWidthOfSpacer(i,Q,c,F,m,h):u;var q=H().horizontalQ(e).scale(g).moveby("category").numberOfObjects(F).objectClass(S).objectSize(c).spacerSize(u).transitionDuration(j).easeFunc(z).namespace(w),W=q.exitFunction();q.exitFunction(function(t){void 0==c&&console.log(t.nodes(),c),W(t),t.selectAll("g").classed("to-remove",!0),t.selectAll("* > rect").transition().duration(j).attr("transform",function(t,e){return"translate(0,"+(C?0:g(B[1]))+")"}).attr("width",e?c:0).attr("height",C?c:0).remove()}),q(L,M,0);var V=[];L.selectAll("g:not(.to-remove)."+S).each(function(t,e){V.push(Number(d3.select(this).attr("parent-index")))}),b="index"==b.colorBy()?b.dataExtent([0,Math.max(...V)]):b.dataExtent(B),L.selectAll("g."+S+":not(.to-remove)").each(function(t,r){var o=d3.select(this),a=(n[t],f(t,r)),r=void 0==o.attr("parent-index")?r:o.attr("parent-index"),i=b(t,a,r,"fill"),l=b(t,a,r,"stroke"),u=gt.sel.safeSelect(o,"rect","bar-rect");void 0==u.attr("transform")&&u.attr("transform",function(t,e){return"translate(0,"+(C?0:g(B[1]))+")"}).attr("width",e?c:0).attr("height",C?c:0),u.transition().duration(j).ease(z).attr("transform",function(t,n){return"translate("+(e?c-c*A:"right"==s?g(B[1])-g(a):c-c*A)+","+(C?c-c*A:g(B[1])-g(a))+")"}).attr("width",e?c*A:g(a)).attr("height",C?c*A:g(a)).attr("fill",i).attr("stroke",l).attr("stroke-width",x),o.on("mouseover",function(t,e){L.selectAll("g."+S).style("opacity",.2),o.style("opacity",1),u.attr("stroke-width",2*x)}),o.on("mouseout",function(){L.selectAll("g."+S).style("opacity",1),u.attr("stroke-width",x)})}),O.selection(L.selectAll(".bar-rect")).data(n),O()}var n,r,o,a,i,l,c,u,s="horizontal",h=!1,f=function(t,e){return n[t]},d=function(t,e){return d3.descending(n[t],n[e])},g=d3.scaleLinear(),p=.5,m=.05,v=50,y=100,x=2,b=I(),k="transparent",w="d3sm-bar",S="bar",j=1e3,z=d3.easeExp,O=Z(),A=1;return e.selection=function(n){return arguments.length?(t=n,e):t},e.data=function(t){return arguments.length?(n=t,e):n},e.orient=function(t){return arguments.length?(s=t,e):s},e.spaceX=function(t){return arguments.length?(r=t,e):r},e.spaceY=function(t){return arguments.length?(o=t,e):o},e.overflowQ=function(t){return arguments.length?(h=t,e):h},e.grouping=function(t){return arguments.length?(a=t,e):a},e.valueExtractor=function(t){return arguments.length?(f=t,e):f},e.sortingFunction=function(t){return arguments.length?(d=t,e):d},e.scale=function(t){return arguments.length?(g=t,e):g},e.domainPadding=function(t){return arguments.length?(p=t,e):p},e.objectSpacer=function(t){return arguments.length?(m=t,e):m},e.minObjectSize=function(t){return arguments.length?(v=t,e):v},e.maxObjectSize=function(t){return arguments.length?(y=t,e):y},e.barStrokeWidth=function(t){return arguments.length?(x=t,e):x},e.colorFunction=function(t){return arguments.length?(b=t,e):b},e.backgroundFill=function(t){return arguments.length?(k=t,e):k},e.namespace=function(t){return arguments.length?(w=t,e):w},e.objectClass=function(t){return arguments.length?(S=t,e):S},e.transitionDuration=function(t){return arguments.length?(j=t,e):j},e.easeFunc=function(t){return arguments.length?(z=t,e):z},e.barKeys=function(t){return arguments.length?(i=t,e):i},e.barValues=function(t){return arguments.length?(l=t,e):l},e.objectSize=function(t){return arguments.length?(c=t,e):c},e.spacerSize=function(t){return arguments.length?(u=t,e):u},e.tooltip=function(t){return arguments.length?(O=t,e):O},e.barPercent=function(t){return arguments.length?(A=t,e):A},e}function $(t){function e(){var e={x:0,y:0,width:r,height:o},g=gt.sel.setupContainer(t,L,e,E);(a=d3.keys(n)).sort(function(t,e){return Q(t,e)||q(t,e)}),gt.con.log("bubbleHeatmap","cells are sorted by",a),i=gt.arr.unique(a.map(y)),l=gt.arr.unique(a.map(x)),c=gt.arr.unique(a.map(b)),u=gt.arr.unique(a.map(k)),gt.con.log("bubbleHeatmap","x and y keys are",{x:i,y:l});var p=i.length,m=l.length,v=[Math.min(...c)-j,Math.max(...c)+j];f=gt.math.calculateWidthOfObject(o,m,O,A,z,w),s=gt.math.calculateWidthOfObject(r,p,O,A,z,w),d=gt.math.calculateWidthOfSpacer(l,o,f,m,z,w),h=gt.math.calculateWidthOfSpacer(i,r,s,p,z,w),gt.con.log("bubbleHeatmap","size of",{x:s,y:f}),S.domain(v).range([Math.min(O/2,Math.min(f,s)/2),Math.min(f,s)/2]);var D=H().horizontalQ(!1).moveby("category").numberOfObjects(m).objectClass(gt.str.hypenate(M,"row")).objectSize(f).spacerSize(d).transitionDuration(F).easeFunc(B).namespace("row"),T=H().horizontalQ(!0).moveby("category").numberOfObjects(p).objectClass(M).objectSize(s).spacerSize(h).transitionDuration(F).easeFunc(B);D(g,l,0),g.selectAll("g."+gt.str.hypenate(M,"row")).each(function(t,e){T(d3.select(this),i,0)});var R=g.selectAll("g:not(.to-remove)."+M).data(a),Y=[];R.each(function(t,e){Y.push(Number(d3.select(this).attr("parent-index")))}),W="index"==W.colorBy()?W.dataExtent([0,Math.max(...Y)]):W.dataExtent([0,Math.max(...u)]),R.each(function(t,e){gt.con.log("bubbleHeatmap","each cell",{key:t,index:e,node:d3.select(this).node()});var r=d3.select(this),o=(n[t],k(t,e)),a=b(t,e),e=void 0==r.attr("parent-index")?e:r.attr("parent-index"),i=W(t,o,e,"fill"),l=W(t,o,e,"stroke");gt.con.log("bubbleHeatmap","radius",{radius:a,scaled:S(a),extent:v,range:S.range()}),gt.sel.safeSelect(r,"circle",gt.str.hypenate(M,"circle")).attr("cx",s/2).attr("cy",f/2).attr("r",S(a)).attr("fill",i).attr("stroke",l).attr("stroke-width",C)}),V.selection(R.selectAll("circle."+gt.str.hypenate(M,"circle"))).data(n),V()}var n,r,o,a,i,l,c,u,s,h,f,d,g="x",p="y",m="r",v="v",y=function(t,e){return n[t][g]},x=function(t,e){return n[t][p]},b=function(t,e){return n[t][m]},k=function(t,e){return n[t][v]},w=!1,S=d3.scaleLinear(),j=.5,z=0,O=50,A=100,C=2,E="transparent",L="d3sm-bubble",M="bubble",F=1e3,B=d3.easeExp,Q=function(t,e){return y(t)-y(e)},q=function(t,e){return x(t)-x(e)},W=I().colorBy("value"),V=Z();return e.selection=function(n){return arguments.length?(t=n,e):t},e.data=function(t){return arguments.length?(n=t,e):n},e.spaceX=function(t){return arguments.length?(r=t,e):r},e.spaceY=function(t){return arguments.length?(o=t,e):o},e.xKey=function(t){return arguments.length?(g=t,e):g},e.yKey=function(t){return arguments.length?(p=t,e):p},e.rKey=function(t){return arguments.length?(m=t,e):m},e.vKey=function(t){return arguments.length?(v=t,e):v},e.cellKeys=function(t){return arguments.length?(a=t,e):a},e.xValues=function(t){return arguments.length?(i=t,e):i},e.yValues=function(t){return arguments.length?(l=t,e):l},e.rValues=function(t){return arguments.length?(c=t,e):c},e.vValues=function(t){return arguments.length?(u=t,e):u},e.xExtractor=function(t){return arguments.length?(y=t,e):y},e.yExtractor=function(t){return arguments.length?(x=t,e):x},e.rExtractor=function(t){return arguments.length?(b=t,e):b},e.vExtractor=function(t){return arguments.length?(k=t,e):k},e.overflowQ=function(t){return arguments.length?(w=t,e):w},e.scale=function(t){return arguments.length?(S=t,e):S},e.domainPadding=function(t){return arguments.length?(j=t,e):j},e.objectSpacer=function(t){return arguments.length?z=t:n},e.minObjectSize=function(t){return arguments.length?(O=t,e):O},e.maxObjectSize=function(t){return arguments.length?(A=t,e):A},e.bubbleStrokeWidth=function(t){return arguments.length?(C=t,e):C},e.backgroundFill=function(t){return arguments.length?(E=t,e):E},e.namespace=function(t){return arguments.length?(L=t,e):L},e.objectClass=function(t){return arguments.length?(M=t,e):M},e.transitionDuration=function(t){return arguments.length?(F=t,e):F},e.easeFunc=function(t){return arguments.length?(B=t,e):B},e.tooltip=function(t){return arguments.length?(V=t,e):V},e.colorFunction=function(t){return arguments.length?(W=t,e):W},e.xSize=function(t){return arguments.length?(s=t,e):s},e.xSpacerSize=function(t){return arguments.length?(h=t,e):h},e.ySize=function(t){return arguments.length?(f=t,e):f},e.ySpacerSize=function(t){return arguments.length?(d=t,e):d},e}function tt(t){function e(){var e={x:0,y:0,width:r,height:o},d=gt.sel.setupContainer(t,A,e,O);a=d3.keys(n),i=gt.arr.unique(a.map(m)),l=gt.arr.unique(a.map(v)),c=gt.arr.unique(a.map(y)),a.sort(function(t,e){return M(t,e)||F(t,e)}),gt.con.log("heatmap","cells are sorted by",a),gt.con.log("heatmap","x and y keys are",{x:i,y:l});var g=i.length,p=l.length;h=gt.math.calculateWidthOfObject(o,p,k,j,b,x),u=gt.math.calculateWidthOfObject(r,g,w,S,b,x),f=gt.math.calculateWidthOfSpacer(l,o,h,p,b,x),s=gt.math.calculateWidthOfSpacer(i,r,u,g,b,x),gt.con.log("heatmap","size of",{x:u,y:h});var q=H().horizontalQ(!1).moveby("category").numberOfObjects(p).objectClass(gt.str.hypenate(C,"row")).objectSize(h+f).spacerSize(0).transitionDuration(E).easeFunc(L).namespace("row"),W=H().horizontalQ(!0).moveby("category").numberOfObjects(g).objectClass(C).objectSize(u+s).spacerSize(0).transitionDuration(E).easeFunc(L);q(d,l,0),d.selectAll("g."+gt.str.hypenate(C,"row")).each(function(t,e){W(d3.select(this),i,0)});var V=d.selectAll("g:not(.to-remove)."+C);if(a.length!=l.length*i.length){var D={};a.map(function(t,e){D[m(t)+"::"+v(t)]=t});for(var T=[],R=0;R.5?l:-l}).attr("cx",function(t,n){var r=a.pointValues[n];if(e){var o=utils.arr.whichBin(a.binned,r),i=Math.random(),l=P(i*a.frequencies[o]*.5);return Math.random()>.5?l:-l}return p(r)}).attr("stroke",function(t,e){var t=a.pointValues[e];return w(t,"stroke",u,O,M)}).attr("fill",function(t,e){var t=a.pointValues[e];return w(t,"fill",u,O,M)}).attr("stroke-width",j),y.selectAll("circle.point").on("mouseover",function(t,e){d.selectAll("g."+A).style("opacity",.2),o.style("opacity",1),h.attr("stroke-width",2*b),g.attr("stroke-width",2*b),d.selectAll(".point").style("opacity",.2),d3.select(this).style("opacity",1).attr("r",2*S).attr("stroke-width",2*j)}),y.selectAll("circle.point").on("mouseout",function(t,e){var n=document.createEvent("SVGEvents");n.initEvent("mouseout",!0,!0),s.node().dispatchEvent(n),d.selectAll(".point").style("opacity",1),d3.select(this).attr("stroke-width",j).attr("r",S)})}else cV.selectAll(".point").transition().duration(C).ease(E).attr("r",0).attr("cy",e?p(R[1])-p(v):P(0)).attr("cx",e?P(0):p(v)).remove()}}),M.selection(d.selectAll("g:not(.to-remove)."+A+" .area")),void 0==M.data()&&M.data(n),M(),void 0==M.values()&&M.values([function(t,e){return t["utils.math.quartiles"][e]},function(t,e){return t["utils.math.quartiles"][e]},function(t,e){return t["utils.math.quartiles"][e]},function(t,e){return t["utils.math.quartiles"][e]},function(t,e){return t["utils.math.quartiles"][e]}])}var n,r,o,a,i,l,c,u,s="horizontal",h=!0,f=!0,d=function(t,e){return n[t]},g=function(t,e){return d3.descending(n[t],n[e])},p=d3.scaleLinear(),m=.5,v=.05,y=50,x=100,b=2,k=I(),w=function(t,e,n,r,o){var a=d3.scaleLinear().domain([r,o]).range([-.25,.05]),i=utils.color.modifyHexidecimalColorLuminance(n.replace("#",""),a(t)),l="stroke"==e?0:.25;return utils.color.modifyHexidecimalColorLuminance(i.replace("#",""),l)},S=3,j=2,z="transparent",O="d3sm-violin",A="violin",C=1e3,E=d3.easeExp,L=["Q0","Q1","Q2","Q3","Q4"],M=Z().keys([L[4],L[3],L[2],L[1],L[0]]),F=Z(),B=function(t,e){return e.points},Q=function(t,e){return e[t].value};return e.violinPointsExtractor=function(t){return arguments.length?(B=t,e):B},e.violinPointValueExtractor=function(t){return arguments.length?(Q=t,e):Q},e.selection=function(n){return arguments.length?(t=n,e):t},e.data=function(t){return arguments.length?(n=t,e):n},e.orient=function(t){return arguments.length?(s=t,e):s},e.spaceX=function(t){return arguments.length?(r=t,e):r},e.spaceY=function(t){return arguments.length?(o=t,e):o},e.overflowQ=function(t){return arguments.length?(h=t,e):h},e.pointsQ=function(t){return arguments.length?(f=t,e):f},e.grouping=function(t){return arguments.length?(a=t,e):a},e.valueExtractor=function(t){return arguments.length?(d=t,e):d},e.sortingFunction=function(t){return arguments.length?(g=t,e):g},e.scale=function(t){return arguments.length?(p=t,e):p},e.domainPadding=function(t){return arguments.length?(m=t,e):m},e.objectSpacer=function(t){return arguments.length?(v=t,e):v},e.minObjectSize=function(t){return arguments.length?(y=t,e):y},e.maxObjectSize=function(t){return arguments.length?(x=t,e):x},e.objectStrokeWidth=function(t){return arguments.length?(b=t,e):b},e.colorFunction=function(t){return arguments.length?(k=t,e):k},e.pointColorFunc=function(t){return arguments.length?(w=t,e):w},e.pointRadius=function(t){return arguments.length?(S=t,e):S},e.pointStrokeWidth=function(t){return arguments.length?(j=t,e):j},e.backgroundFill=function(t){return arguments.length?(z=t,e):z},e.namespace=function(t){return arguments.length?(O=t,e):O},e.objectClass=function(t){return arguments.length?(A=t,e):A},e.transitionDuration=function(t){return arguments.length?(C=t,e):C},e.easeFunc=function(t){return arguments.length?(E=t,e):E},e.quartileKey=function(t){return arguments.length?(quartileKey=t,e):quartileKey},e.quartileKeys=function(t){return arguments.length?(L=t,e):L},e.violinKeys=function(t){return arguments.length?(i=t,e):i},e.violinValues=function(t){return arguments.length?(l=t,e):l},e.objectSize=function(t){return arguments.length?(c=t,e):c},e.spacerSize=function(t){return arguments.length?(u=t,e):u},e.tooltip=function(t){return arguments.length?(M=t,e):M},e.pointsTooltip=function(t){return arguments.length?(F=t,e):F},e}function nt(){function t(t,a){var i=a[t],l=e(t,i),c=d3.keys(l),u=c.map(function(t,e){return n(t,l)}),s=utils.math.quartiles(u,o),h=d3.histogram()(u),f=h.map(t=>t.length),d=r?{x:0,y:d3.min(u)}:{x:d3.min(u),y:0},g=r?{x:0,y:d3.max(u)}:{x:d3.max(u),y:0},p=h.map(function(t,e){return r?{y:t.length?d3.median(t):d3.median([t.x0,t.x1]),x:f[e]}:{x:t.length?d3.median(t):d3.median([t.x0,t.x1]),y:f[e]}});p=[d].concat(p).concat([g]),i.binned=h,i.frequencies=f,i.contour=p,i.utils.math.quartiles=s,i.pointKeys=c,i.pointValues=u}var e,n,r=!0,o=["Q0","Q1","Q2","Q3","Q4"];return t.horizontalQ=function(e){return arguments.length?(r=e,t):r},t.quartileKeys=function(e){return arguments.length?(o=e,t):o},t.violinPointsExtractor=function(n){return arguments.length?(e=n,t):e},t.violinPointValueExtractor=function(e){return arguments.length?(n=e,t):n},t}function rt(t){function e(){var e="horizontal"==p,n=!e,r={x:0,y:0,width:a,height:i},x=gt.sel.setupContainer(t,k,r,b);l=d3.keys(o),c=gt.arr.unique(l.map(z)).sort(),u=gt.arr.unique(l.map(O)).sort().sort(function(t,e){return t.split(";").length-e.split(";").length}),e?l.sort(function(t,e){return M(t,e)||L(t,e)}):l.sort(function(t,e){return L(t,e)||M(t,e)});var A=e?u:c,F=e?c:u,B=e?A.length:F.length,Q=e?F.length:A.length;d=gt.math.calculateWidthOfObject(a,B,v,y,C,m),h=gt.math.calculateWidthOfObject(i,Q,v,y,E,m),g=gt.math.calculateWidthOfSpacer(A,a,d,B,C,m),f=gt.math.calculateWidthOfSpacer(F,i,h,Q,E,m);var q=H().horizontalQ(!1).moveby("category").numberOfObjects(Q).objectSize(h).spacerSize(f).transitionDuration(S).easeFunc(j),W=H().horizontalQ(!0).moveby("category").numberOfObjects(B).objectClass(w).objectSize(d).spacerSize(g).transitionDuration(S).easeFunc(j);n?(W.objectClass(w),q.namespace("across").objectClass(gt.str.hypenate(w,"across")),q(x,F,0),x.selectAll("g."+gt.str.hypenate(w,"across")).each(function(t,e){W(d3.select(this),A,0)})):(W.namespace("across").objectClass(gt.str.hypenate(w,"across")),q.objectClass(w),W(x,A,0),x.selectAll("g."+gt.str.hypenate(w,"across")).each(function(t,e){q(d3.select(this),F,0)}));var V=x.selectAll("g:not(.to-remove)."+w),D={};l.map(function(t,e){D[z(t)+"::"+O(t)]=t}),V.data(l),V.each(function(t,e){var n=d3.select(this);if(void 0!=t){o[t];var r=z(t,e),a=O(t,e);n.classed(a,!0),n.classed(r,!0),gt.sel.safeSelect(n,"circle",gt.str.hypenate(w,"circle")).attr("cx",d/2).attr("cy",h/2).attr("r",void 0==s?Math.min(d,h)/2:s).attr("fill",a.includes(r)?"black":"rgb(233,233,233)").attr("stroke","black").attr("in-intersection",a.includes(r))}})}function n(){var t={};return u.map(function(e,n){t[e]={total:0}}),l.map(function(e,n){var r=A(e,n);0==t[O(e,n)].total&&(Array.isArray(r)?(t[O(e,n)].total+=r.length,t[O(e,n)].values=r):t[O(e,n)].total+=r)}),t}function r(){var t={};return c.map(function(e,n){t[e]={total:0}}),l.map(function(e,n){var r=A(e,n);Array.isArray(r)?t[z(e,n)].total+=r.length:t[z(e,n)].total+=r}),t}var o,a,i,l,c,u,s,h,f,d,g,p="horizontal",m=!1,v=20,y=50,x=2,b="transparent",k="d3sm-upset",w="upset",S=1e3,j=d3.easeExp,z=function(t,e){return o[t].set},O=function(t,e){return o[t].intersection},A=function(t,e){return o[t].elements},C=.05,E=.05,L=function(t,e){return c.indexOf(z(t))-c.indexOf(z(e))},M=function(t,e){return u.indexOf(O(t))-u.indexOf(O(e))};return e.selection=function(n){return arguments.length?(t=n,e):t},e.data=function(t){return arguments.length?(o=t,e):o},e.orient=function(t){return arguments.length?(p=t,e):p},e.spaceX=function(t){return arguments.length?(a=t,e):a},e.spaceY=function(t){return arguments.length?(i=t,e):i},e.overflowQ=function(t){return arguments.length?(m=t,e):m},e.minObjectSize=function(t){return arguments.length?(v=t,e):v},e.maxObjectSize=function(t){return arguments.length?(y=t,e):y},e.circleStrokeWidth=function(t){return arguments.length?(x=t,e):x},e.backgroundFill=function(t){return arguments.length?(b=t,e):b},e.namespace=function(t){return arguments.length?(k=t,e):k},e.objectClass=function(t){return arguments.length?(w=t,e):w},e.transitionDuration=function(t){return arguments.length?(S=t,e):S},e.easeFunc=function(t){return arguments.length?(j=t,e):j},e.cellKeys=function(t){return arguments.length?(l=t,e):l},e.setValues=function(t){return arguments.length?(c=t,e):c},e.intersectionValues=function(t){return arguments.length?(u=t,e):u},e.xObjectSpacer=function(t){return arguments.length?(C=t,e):C},e.yObjectSpacer=function(t){return arguments.length?(E=t,e):E},e.radius=function(t){return arguments.length?(s=t,e):s},e.setExtractor=function(t){return arguments.length?(z=t,e):z},e.intersectionExtractor=function(t){return arguments.length?(O=t,e):O},e.elementExtractor=function(t){return arguments.length?(A=t,e):A},e.setKeySortingFunction=function(t){return arguments.length?(L=t,e):L},e.intersectionKeySortingFunction=function(t){return arguments.length?(M=t,e):M},e.yObjectSize=function(t){return arguments.length?(h=t,e):h},e.ySpacerSize=function(t){return arguments.length?(f=t,e):f},e.xObjectSize=function(t){return arguments.length?(d=t,e):d},e.xSpacerSize=function(t){return arguments.length?(g=t,e):g},e.intersectionTotals=n,e.setTotals=r,e}function ot(t){function e(){var e="horizontal"==l,r=!e,u={x:0,y:0,width:o,height:a},k=gt.sel.setupContainer(t,v,u,m);p.dataExtent([0,n.length-1]).colorBy("categories").categoryExtractor(function(t,e,n){return e});var w=Math.min(o,a)/2,S=n.length,j=void 0==i?n.sort(s):i,z=gt.arr.flatten(j),O=e?o:a,A=gt.math.calculateWidthOfObject(O,S,f,d,h,c),C=gt.math.calculateWidthOfSpacer(z,O,A,S,h,c);H().horizontalQ(e).scale(scale).moveby("category").numberOfObjects(S).objectClass(y).objectSize(A).spacerSize(C).transitionDuration(x).easeFunc(b).namespace(v)(k,j,0);w=Math.min(A,o,a)/2-g;k.selectAll("g:not(.to-remove)."+y).each(function(t,n){var a=d3.select(this),i=gt.sel.safeSelect(a,"circle"),l=p(void 0,t,n,"fill"),c=p(void 0,t,n,"stroke"),u=e?w+g:(o-2*w)/2+w,s=r?w+g:(o-2*w)/2+w;i.attr("r",w).attr("cx",u).attr("cy",s).attr("fill",l).attr("stroke",c).attr("stroke-width",g);var h=gt.sel.safeSelect(a,"text");h.text(t).attr("text-anchor","middle").attr("transform",function(t,e){return"translate("+u+","+(s+h.node().getBoundingClientRect().height/4)+")"})})}var n,r,o,a,i,l="horizontal",c=!1,u=function(t,e){return r[t]},s=function(t,e){return d3.ascending(t,e)},h=.05,f=10,d=100,g=2,p=I(),m="transparent",v="d3sm-legend",y="legend",x=1e3,b=d3.easeExp;return e.categories=function(t){return arguments.length?(n=t,e):n},e.selection=function(n){return arguments.length?(t=n,e):t},e.data=function(t){return arguments.length?(r=t,e):r},e.orient=function(t){return arguments.length?(l=t,e):l},e.spaceX=function(t){return arguments.length?(o=t,e):o},e.spaceY=function(t){return arguments.length?(a=t,e):a},e.overflowQ=function(t){return arguments.length?(c=t,e):c},e.grouping=function(t){return arguments.length?(i=t,e):i},e.valueExtractor=function(t){return arguments.length?(u=t,e):u},e.sortingFunction=function(t){return arguments.length?(s=t,e):s},e.objectSpacer=function(t){return arguments.length?(h=t,e):h},e.minObjectSize=function(t){return arguments.length?(f=t,e):f},e.maxObjectSize=function(t){return arguments.length?(d=t,e):d},e.bubbleStrokeWidth=function(t){return arguments.length?(g=t,e):g},e.colorFunction=function(t){return arguments.length?(p=t,e):p},e.backgroundFill=function(t){return arguments.length?(m=t,e):m},e.namespace=function(t){return arguments.length?(v=t,e):v},e.objectClass=function(t){return arguments.length?(y=t,e):y},e.transitionDuration=function(t){return arguments.length?(x=t,e):x},e.easeFunc=function(t){return arguments.length?(b=t,e):b},e}function at(t){function e(){var e={x:0,y:0,width:r,height:o},h=gt.sel.setupContainer(t,c,e,s),f=gt.sel.safeSelect(t,"defs"),d=gt.sel.safeSelect(f,"linearGradient").attr("x1","0%").attr("y1","100%").attr("x2","0%").attr("y2","0%").attr("id",gt.str.hypenate(c,"numerical-legend-gradient"));l.dataExtent([a,i]).colorBy("value").valueExtractor(function(t,e,n){return e}),d.selectAll("stop").data(l.colors()).enter().append("stop").attr("offset",function(t,e){return e/(l.colors().length-1)}).attr("stop-color",function(t){return t});var g=gt.sel.safeSelect(h,"rect","legend").attr("transform","translate(0,"+u+")").style("fill","url(#"+gt.str.hypenate(c,"numerical-legend-gradient")+")").attr("x",0).attr("y",0).attr("width",r).attr("height",o-2*u).on("mousemove",function(t,e){n(t,e,g,d3.select(this))}).on("mouseout",function(t,e){d3.select("#"+gt.str.hypenate(c,"legend-tooltip")).remove()});gt.sel.safeSelect(h,"text","min").text(gt.math.round(a,2)).attr("text-anchor","middle").attr("font-size",u+"px").attr("transform",function(t,e){return"translate("+r/2+","+(o-u/4)+")"}),gt.sel.safeSelect(h,"text","max").text(gt.math.round(i,2)).attr("text-anchor","middle").attr("font-size",u+"px").attr("transform",function(t,e){return"translate("+r/2+","+u+")"})}function n(t,e,n,r){var o=d3.scaleLinear().domain([0,n.attr("height")]).range([i,a]),s=d3.mouse(n.node()),d=gt.math.round(o(s[1]),f),g=l(void 0,d,void 0,"stroke"),p=l(void 0,d,void 0,"fill"),m=gt.sel.safeSelect(d3.select("body"),"div",gt.str.hypenate(c,"legend-tooltip")).attr("id",gt.str.hypenate(c,"legend-tooltip")).style("position","absolute").style("left",d3.event.pageX+15+"px").style("top",d3.event.pageY+15+"px").style("background-color",p).style("border-color",g).style("min-width",u*(String(i).split(".")[0].length+3)+"px").style("min-height",u*(String(i).split(".")[0].length+3)+"px").style("border-radius","50%").style("border-radius","5000px").style("display","flex").style("justify-content","center").style("text-align","middle").style("padding","2px").style("border-style","solid").style("border-width",2);gt.sel.safeSelect(m,"div").text(d).style("color",h).style("align-self","center")}var r,o,a=0,i=1,l=I(),c="d3sm-linear-vertical-gradient",u=12,s="transparent",h="black",f=2;return e.min=function(t){return arguments.length?(a=t,e):a},e.max=function(t){return arguments.length?(i=t,e):i},e.spaceX=function(t){return arguments.length?(r=t,e):r},e.spaceY=function(t){return arguments.length?(o=t,e):o},e.namespace=function(t){return arguments.length?(c=t,e):c},e.fontSize=function(t){return arguments.length?(u=t,e):u},e.backgroundFill=function(t){return arguments.length?(s=t,e):s},e.colorFunction=function(t){return arguments.length?(l=t,e):l},e.textColor=function(t){return arguments.length?(h=t,e):h},e.roundTo=function(t){return arguments.length?(f=t,e):f},e}function it(t){function e(){z&&i()}function n(t){z=void 0!=t?t:!z,y=gt.sel.getTranslation(v),x=gt.sel.getTranslation(p),z?g.node().addEventListener("mousedown",a,!0):(g.node().removeEventListener("mousedown",a,!0),o())}function r(){y=gt.sel.getTranslation(v),x=gt.sel.getTranslation(p);var t=gt.sel.safeSelect(p,"g","lasso-container").selectAll('path[instance="'+E+'"]'),e=((t=t.data(A)).exit().remove(),t.enter().append("path"));l(t=t.merge(e))}function o(){var t=gt.sel.safeSelect(p,"g","lasso-container");t.selectAll('path[instance="'+E+'"]').remove();t.remove(),p.selectAll(m).classed("in-lasso",!1),h()}function a(t){t.preventDefault(),t.stopPropagation();var e=gt.sel.safeSelect(p,"g","lasso-container");O=[],g.node().addEventListener("mousemove",c),g.node().addEventListener("mouseup",function(t){g.node().removeEventListener("mousemove",c),A.push(O),A=gt.arr.unique(A)}),l(S=e.append("path").data([O]))}function i(){var t=gt.sel.safeSelect(p,"g","lasso-container").selectAll('path[instance="'+E+'"]'),e=((t=t.data(A)).exit().remove(),t.enter().append("path"));l(t=t.merge(e).transition().duration(R).ease(Y))}function l(t){t.attr("class",gt.str.hypenate(j,"lasso-path")).style("opacity",B).attr("fill",M).attr("d",C).attr("instance",E).style("stroke-dasharray",Q).attr("stroke",q).attr("stroke-width",W).style("animation","lassoDash "+F+" linear").style("animation-iteration-count","infinite")}function c(t){if(void 0!=b&&b.dispatch(gt.str.hypenate(j,"drag")),1==t.which){d3.event=t;var e=d3.mouse(p.node()),e=d3.mouse(g.node());if(void 0!=k&&(e[0]=k.invert(e[0])),void 0!=w&&(e[1]=w.invert(e[1])),e[0]=e[0]-y[0]-x[0],e[1]=e[1]-y[1]-x[1],O.length){var n=O[O.length-1],r=[e[0],e[1]],o=[n[0],n[1]];k&&(o[0]=k(o[0]),r[0]=k(r[0])),w&&(o[1]=w(o[1]),r[1]=w(r[1])),gt.math.euclideanDistance(o,r)>L&&u(e)}else u(e)}}function u(t){if(void 0!=t){if(O.push(t),S.attr("d",C),O.length<3)return;s(A.concat([O]))}else s(A)}function s(t){return void 0==t&&(t=A),p.selectAll(m).each(function(e,n){var r=d3.select(this),o=r.absolutePosition(),a=[[o.left-y[0]-x[0],o.top-y[1]-x[1]],[o.right-y[0]-x[0],o.top-y[1]-x[1]],[o.left-y[0]-x[0],o.bottom-y[1]-x[1]],[o.right-y[0]-x[0],o.bottom-y[1]-x[1]]];void 0!=k&&(a[0][0]=k.invert(a[0][0]),a[1][0]=k.invert(a[1][0]),a[2][0]=k.invert(a[2][0]),a[3][0]=k.invert(a[3][0])),void 0!=w&&(a[0][1]=w.invert(a[0][1]),a[1][1]=w.invert(a[1][1]),a[2][1]=w.invert(a[2][1]),a[3][1]=w.invert(a[3][1]));for(var i=!1,n=0;nd3.polygonContains(l,t))&&(i=!0)}r.classed("in-lasso",i),r.classed("in-lasso-"+E,i)}),h(),p.selectAll(".in-lasso-"+E)}function h(){p.selectAll(m).each(function(t,e){var n=d3.select(this);f(n,n.classed("in-lasso"))})}function f(t,e){var n=t.attr("_pre_lasso_fill"),r=t.attr("_pre_lasso_stroke"),o=t.attr("_pre_lasso_stroke-width");e?(t.classed("in-lasso",!0),t.classed("in-lasso-"+E,!0),void 0==n&&t.attr("_pre_lasso_fill",t.attr("fill")),void 0==r&&t.attr("_pre_lasso_stroke",t.attr("stroke")),void 0==o&&t.attr("_pre_lasso_stroke-width",t.attr("stroke-width")),t.attr("fill",V).attr("stroke",D).attr("stoke-width",T)):(t.classed("in-lasso",!1),t.classed("in-lasso-"+E,!1),void 0!=n&&t.attr("fill",n),void 0!=r&&t.attr("stroke",r),void 0!=o&&t.attr("stroke-width",o))}function d(){d3.select("html").select("style."+gt.str.hypenate(j,"lasso-dash")).empty()&&d3.select("html").append("style").classed(gt.str.hypenate(j,"lasso-dash"),!0).html("@keyframes lassoDash {to { stroke-dashoffset: 1000;}}")}var g,p,m,v,y,x,b,k,w,S,j="d3sm-lasso",z=!1,O=[],A=[],C=d3.line().x(function(t,e){return void 0!=k?k(t[0]):t[0]}).y(function(t,e){return void 0!=w?w(t[1]):t[1]}).curve(d3.curveLinearClosed),E=0,L=10,M="#17a2b8",F="10s",B=.3,Q="5, 10",q="black",W=2,V="white",D="black",T=3,R=1e3,Y=d3.easeExp;return e.svg=function(t){return arguments.length?(g=t,e):g},e.chartContainer=function(t){return arguments.length?(v=t,e):v},e.objectContainer=function(t){return arguments.length?(p=t,e):p},e.objectClass=function(t){return arguments.length?(m=t,e):m},e.namespace=function(t){return arguments.length?(j=t,e):j},e.xScale=function(t){return arguments.length?(k=t,e):k},e.yScale=function(t){return arguments.length?(w=t,e):w},e.activeQ=function(t){return arguments.length?(z=t,e):z},e.currentPoints=function(t){return arguments.length?(O=t,e):O},e.allPoints=function(t){return arguments.length?(A=t,e):A},e.instance=function(t){return arguments.length?(E=t,e):E},e.tickDistance=function(t){return arguments.length?(L=t,e):L},e.color=function(t){return arguments.length?(M=t,e):M},e.animationRate=function(t){return arguments.length?(F=t,e):F},e.opacity=function(t){return arguments.length?(B=t,e):B},e.dashArray=function(t){return arguments.length?(Q=t,e):Q},e.stroke=function(t){return arguments.length?(q=t,e):q},e.lassoedFill=function(t){return arguments.length?(V=t,e):V},e.lassoedStroke=function(t){return arguments.length?(D=t,e):D},e.lassoedStrokeWidth=function(t){return arguments.length?(T=t,e):T},e.eventCatcher=function(t){return arguments.length?(b=t,e):b},e.drag=c,e.draw=r,e.tick=u,e.detect=s,e.toggle=n,e.remove=o,e.render=a,e.keyFrames=d,e.updateObjects=h,e.applyPathAttributes=l,e.applyObjectAttributes=f,d(),e}function lt(t){function e(){var e=c.select("."+gt.str.hypenate(t.namespace(),"object-container")),n=gt.sel.getTranslation(e.attr("transform")),r=e.attr("transform","translate(0,0)");i=c.node().getBBox().width-t.spaceX(),l=c.node().getBBox().height-t.spaceY(),r.attr("transform","translate("+n[0]+","+n[1]+")"),gt.con.log("plotZoom","setLocks",{xLock:i,yLock:l})}function n(){e();var n,h,f=u.map(function(t,e){return t.selection()}),d=s.map(function(t,e){return t.selection()});"2D"==a&&(n=!0,h=!0),"horizontal"==a&&(n=!0,h=!1),"vertical"==a&&(h=!0,n=!1);var g=d3.event.transform,p=c.node().getBBox();f.map(function(t,e){return t.node().getBBox()}),f.map(function(t,e){return t.node().getBBox()}),p.width,p.x,p.height,p.y;if("wheel"==r){d3.event.preventDefault();var m=d3.event.deltaY*o,v=d3.event.shiftKey;(g="2D"==a?v?{k:1,x:m,y:0}:{k:1,x:0,y:m}:n?{k:1,x:m,y:0}:{k:1,x:0,y:m}).applyX=function(t){return t*this.k+-1*this.x},g.applyY=function(t){return t*this.k+-1*this.y}}var y=c.select("."+gt.str.hypenate(t.namespace(),"object-container")),x=f.map(function(t,e){return t.select("."+gt.str.hypenate(u[e].namespace(),"object-container"))}),b=d.map(function(t,e){return t.select("."+gt.str.hypenate(s[e].namespace(),"object-container"))}),k=gt.sel.getTranslation(y.attr("transform")),w=(x.map(function(t,e){return gt.sel.getTranslation(t.attr("transform"))}),b.map(function(t,e){return gt.sel.getTranslation(t.attr("transform"))}),n?g.applyX(k[0]):0);n&&(w=w<-i?(g.x=0,-i):(g.x=0,Math.min(w,0)));var S=h?g.applyY(k[1]):0;h&&(S=S<-l?(g.y=0,-l):(g.y=0,Math.min(S,0))),y.attr("transform","translate("+w+","+S+")"),n&&x.map(function(t,e){t.attr("transform","translate("+w+",0)")}),h&&b.map(function(t,e){t.attr("transform","translate(0,"+S+")")})}var r,o=20,a=void 0==t.orient?"horizontal":t.orient(),i=t.spaceX(),l=t.spaceY(),c=t.selection(),u=(d3.select(c.thisSVG()),[]),s=[];return n.eventType=function(t){return arguments.length?(r=t,n):r},n.wheelSpeed=function(t){return arguments.length?(o=t,n):o},n.orient=function(t){return arguments.length?(a=t,n):a},n.xLock=function(t){return arguments.length?(i=t,n):i},n.yLock=function(t){return arguments.length?(l=t,n):l},n.xComponents=function(t){return arguments.length?(u=t,n):u},n.yComponents=function(t){return arguments.length?(s=t,n):s},n.setLocks=e,n.reset=function(){var e=c.select("."+gt.str.hypenate(t.namespace(),"object-container")),n=xAxisSel.select("."+gt.str.hypenate(xAxis.namespace(),"object-container")),r=yAxisSel.select("."+gt.str.hypenate(yAxis.namespace(),"object-container"));e.attr("transform","translate(0,0)"),n.attr("transform","translate(0,0)"),r.attr("transform","translate(0,0)")},n}function ct(t,e,n){function r(){var e=s.select("."+gt.str.hypenate(t.namespace(),"object-container")),n=gt.sel.getTranslation(e.attr("transform")),r=e.attr("transform","translate(0,0)");c=s.node().getBBox().width-.9*t.spaceX(),u=s.node().getBBox().height-.9*t.spaceY(),r.attr("transform","translate("+n[0]+","+n[1]+")"),gt.con.log("plotZoom","setLocks",{xLock:c,yLock:u})}function o(){r();var o,d;"2D"==l&&(o=!0,d=!0),"horizontal"==l&&(o=!0,d=!1),"vertical"==l&&(d=!0,o=!1);var g=d3.event.transform,p=s.node().getBBox(),m=h.node().getBBox(),v=h.node().getBBox();p.width,p.x,p.height,p.y,m.width,m.height,v.width,v.height;if("wheel"==a){d3.event.preventDefault();var y=d3.event.deltaY*i,x=d3.event.shiftKey;(g="2D"==l?x?{k:1,x:y,y:0}:{k:1,x:0,y:y}:o?{k:1,x:y,y:0}:{k:1,x:0,y:y}).applyX=function(t){return t*this.k+-1*this.x},g.applyY=function(t){return t*this.k+-1*this.y}}var b=s.select("."+gt.str.hypenate(t.namespace(),"object-container")),k=h.select("."+gt.str.hypenate(e.namespace(),"object-container")),w=f.select("."+gt.str.hypenate(n.namespace(),"object-container")),S=gt.sel.getTranslation(b.attr("transform")),j=(gt.sel.getTranslation(k.attr("transform")),gt.sel.getTranslation(w.attr("transform")),o?g.applyX(S[0]):0);o&&(j=j<-c?(g.x=0,-c):(g.x=0,Math.min(j,0)));var z=d?g.applyY(S[1]):0;d&&(z=z<-u?(g.y=0,-u):(g.y=0,Math.min(z,0))),b.attr("transform","translate("+j+","+z+")"),o&&k.attr("transform","translate("+j+",0)"),d&&w.attr("transform","translate(0,"+z+")")}var a,i=20,l=void 0==t.orient?"horizontal":t.orient(),c=t.spaceX(),u=t.spaceY(),s=t.selection(),h=e.selection(),f=n.selection();d3.select(s.thisSVG());return o.eventType=function(t){return arguments.length?(a=t,o):a},o.wheelSpeed=function(t){return arguments.length?(i=t,o):i},o.orient=function(t){return arguments.length?(l=t,o):l},o.xLock=function(t){return arguments.length?(c=t,o):c},o.yLock=function(t){return arguments.length?(u=t,o):u},o.setLocks=r,o.reset=function(){var r=s.select("."+gt.str.hypenate(t.namespace(),"object-container")),o=h.select("."+gt.str.hypenate(e.namespace(),"object-container")),a=f.select("."+gt.str.hypenate(n.namespace(),"object-container"));r.attr("transform","translate(0,0)"),o.attr("transform","translate(0,0)"),a.attr("transform","translate(0,0)")},o}var ut=Object.freeze({uniqueElements:e,all:n,tally:r,hasQ:o,first:a,last:i,total:l,unique:c,get:u,listOfListsQ:s,cut:h,groupBy:f,arrayEquals:d,elementsAtLevels:g,numberOfElements:p,flatten:m,whichBin:v}),st=Object.freeze({modifyHexidecimalColorLuminance:y,interpolateColors:x}),ht=Object.freeze({round:b,tickRange:k,partitionRangeInto:w,euclideanDistance:S,quartiles:j,calculateWidthOfObject:z,calculateWidthOfSpacer:O,spacersNeededAtEachLevel:A}),ft=Object.freeze({resizeDebounce:C,setupStandardChartContainers:L}),dt=Object.freeze({hypenate:M,truncateText:F,truncateString:B});let gt={arr:ut,color:st,math:ht,misc:ft,sel:Object.freeze({getTranslation:Q,getContainingSVG:q,safeSelect:W,cpRect:V,bgRect:D,setupContainer:T}),str:dt,con:Object.freeze({consoleGroup:R,consoleGroupEnd:Y,log:K,warn:X,info:P,error:_}),paths:Object.freeze({whiskerPath:N})},pt={scatter:J,bar:U,bubble:$,heatmap:tt,violin:et,neededViolinValues:nt,upset:rt},mt={categorical:ot,numeric:at};d3.selection.prototype.thisSVG=function(){return gt.sel.getContainingSVG(this.node())},d3.mouse.absolute=function(){var t=d3.select("html").node(),[e,n]=this(t);return[e,n]},d3.selection.prototype.absolutePosition=function(){var t=this.node(),e=t.getBoundingClientRect(),n=getContainingSVG(t).getBoundingClientRect();return{top:e.top-n.top,left:e.left-n.left,bottom:e.bottom-n.top,right:e.right-n.left,height:e.height,width:e.width}},d3.selection.prototype.relativePositionTo=function(t){var e=this.node().getBoundingClientRect(),n=t.getBoundingClientRect();return{top:e.top-n.top,left:e.left-n.left,bottom:e.bottom-n.top,right:e.right-n.left,height:e.height,width:e.width}};let vt={aux:{lasso:it,plotZoom:ct,multiPlotZoom:lt},axis:G,charts:pt,colorFunction:I,debugQ:!1,groupingSpacer:H,legends:mt,tooltip:Z,utils:gt};"undefined"!=typeof window&&(window.d3sm=vt),t.default=vt}(this.d3sm=this.d3sm||{}); +!function(t){"use strict";function e(t,e,n){return n.indexOf(t)===e}function n(t,e){return void 0==e?t.every(function(t){return!0===t}):t.every(function(t){return e(t)})}function r(t){var e={};return t.map(function(t){a(Object.keys(e),t)?e[t]=1:e[t]+=1}),e}function a(t,e){return t.includes(e)}function o(t){return t[0]}function i(t){return t[t.length-1]}function l(t){return t.reduce((t,e)=>t+e,0)}function c(t){return t.filter(e)}function s(t,e){return t.filter(function(t,n){return a(e,n)})}function u(t){return n(t.map(function(t,e){return Array.isArray(t)}))}function f(t,e){return u(t)?e.map(function(e,n){return t.get(e)}):s(t,e)}function d(t,e,n){void 0==n&&(n=c(t.map(function(t,n){return element[e]}))).map(function(t,e){r[t]=[]});var r={};return t.map(function(t,n){r[t[e]].push(t)}),r}function h(t,e){if(!e)return!1;if(t.length!=e.length)return!1;for(var n=0,r=t.length;n=n.length?n.push(t.length):n[e]+=t.length,t.map(function(t,r){Array.isArray(t)&&g(t,e,n)}),n}function p(t,e){return e=void 0==e?0:e,t.map(function(t,n){Array.isArray(t)?e=p(t,e):e+=1}),e}function m(t,e){return e=void 0==e?[]:e,t.map(function(t,n){Array.isArray(t)?e=e.concat(m(t)):e.push(t)}),e}function v(t,e){for(var n=0;ntt>n),o=void 0==(o=d3.median(r))?n:o,i=void 0==(i=d3.min(r))?o:i,l=void 0==(l=d3.median(a))?n:l,c=void 0==(c=d3.max(a))?l:c,s="q0",u="q1",f="q2",d="q3",h="q4",g={};return void 0!=e&&5==e.length&&(s=e[0],u=e[1],f=e[2],d=e[3],h=e[4]),g[s]=i,g[u]=o,g[f]=n,g[d]=l,g[h]=c,g}function z(t,e,n,r,a,o){var i=t-(e-1)*(a=0==a||a>1?a:t*a),l=(i=i<0?0:i)/e;return o&&void 0!=n&&l=n.length?n.push(t.length-1):n[e]+=t.length-1,t.map(function(t,r){Array.isArray(t)&&A(t,e,n)}),n}function C(){return Array.prototype.slice.call(arguments).join("-")}function E(t,e,n,r,a,o){var i=t.node().getBoundingClientRect();for(t.text(e);Math.max(i.width,i.height)>a-r&&(e=String(e),e=e.slice(0,e.length-1),t.text(e+"..."),i=t.node().getBoundingClientRect(),0!=e.length););}function L(t,e,n){var r=e/n;return r g.'+l+"."+u);"function"==typeof d?v.each(function(t,e){d(d3.select(this))}):v.remove()}else{x+=n;var m=h.select("g."+u+'[level="'+g+'"] > g.'+l+"."+u);m.empty()&&(m=h.append("g").attr("class",l).classed(u,!0)),m.attr("parent-index",r);var v=h.selectAll("g."+u+'[level="'+(g+1)+'"]');"function"==typeof d?v.each(function(t,e){d(d3.select(this))}):v.remove()}x+=r==p.size()-1?0:y}),x}var e,n,r,a=!0,o=d3.scaleLinear(),i="category",l="d3sm-groupped-item",c=1e3,s=d3.easeSin,u="spacer",f=function(t){t.attr("transform",function(t,e){return"translate("+(a?window.outerWidth:0)+","+(a?0:window.outerWidth)+")"})},d=function(t){R("groupingSpacer","exiting with",{current:t,currentNode:t.node()}),t.selectAll("g").classed("to-remove",!0),t.transition().duration(.9*c).ease(s).attr("transform",function(t,e){return"translate("+(a?window.outerWidth:0)+","+(a?0:window.outerWidth)+")"}).remove()};return t.horizontalQ=function(e){return arguments.length?(a=e,t):a},t.scale=function(e){return arguments.length?(o=e,t):o},t.moveby=function(e){return arguments.length?(i=e,t):i},t.numberOfObjects=function(n){return arguments.length?(e=n,t):e},t.objectClass=function(e){return arguments.length?(l=e,t):l},t.objectSize=function(e){return arguments.length?(n=e,t):n},t.spacerSize=function(e){return arguments.length?(r=e,t):r},t.transitionDuration=function(e){return arguments.length?(c=e,t):c},t.easeFunc=function(e){return arguments.length?(s=e,t):s},t.namespace=function(e){return arguments.length?(u=e,t):u},t.enterFunction=function(e){return arguments.length?(f=e,t):f},t.exitFunction=function(e){return arguments.length?(d=e,t):d},t}function H(t){function e(){function e(t,e,n,r,a){return n&&r?a/2:0}function W(t,e,n,r,a){return n&&r?a/2:0}var D=!!xt.arr.hasQ(["top","bottom","horizontal"],h),_=!D,H={x:0,y:0,width:g,height:p};"left"==h&&(H.x-=g,y&&(H.width+=s),H.y-=V,H.height+=2*V),"bottom"==h&&(H.y=H.y,y&&(H.y-=s,H.height+=s),H.x-=V,H.width+=2*V),"top"==h&&(H.y-=p,y&&(H.height+=s),H.y-=V,H.height+=2*V),"right"==h&&(H.x=0,y&&(H.width+=s,H.x-=s),H.y-=V,H.height+=2*V);var Z=xt.sel.setupContainer(t,z,H,j);"top"==h&&(l=void 0==l?"start":l,c=void 0==c?-90:c),"bottom"==h&&(l=void 0==l?"end":l,c=void 0==c?-90:c),"left"==h&&(l=void 0==l?"end":l,c=void 0==c?0:c),"right"==h&&(l=void 0==l?"start":l,c=void 0==c?0:c);var I=v?void 0==a?o:a:void 0==a?void 0!=A?xt.math.tickRange(...d3.extent(i),A):i:a,J=xt.arr.flatten(I),U=J.length,$=D?g:p,tt=d3.extent(J);N&&tt.reverse();var et=N?[tt[0]+b,tt[1]-b]:[tt[0]-b,tt[1]+b];x.domain(et).range([D?0:p,D?g:0]),u=void 0==u?xt.math.calculateWidthOfObject($,U,k,w,S,m):u,f=void 0==f?xt.math.calculateWidthOfSpacer(J,$,u,U,S,m):f;var nt=xt.str.hypenate(z,v?O+"-categorical":O),rt=G().horizontalQ(D).scale(x).moveby(v?"category":"scale").numberOfObjects(U).objectClass(nt).objectSize(u).spacerSize(f).transitionDuration(Y).easeFunc(X).namespace(z),at=function(t){var e=x(t.datum()),n=2*x(tt[1]),r=ewindow.innerWidth&&a.style("left",d3.event.pageX-15-300+"px")}function r(t,e){var n=d3.select(this).style("fill","black");if(d3.select(n.node().parentNode).select("line."+xt.str.hypenate(z,"tick")).attr("stroke",L).attr("stroke-width",F),y){var r=d3.select(n.node().parentNode).select("line."+xt.str.hypenate(z,"guideline")),a=r.attr("minor");r.attr("stroke",function(t,e){return"true"==a?xt.color.modifyHexidecimalColorLuminance(P,.8):P}).attr("stroke-width",function(t,e){return"true"==a?.8*R:R})}d3.select("#"+xt.str.hypenate(z,"guideline-tooltip")).remove()}var a,o,i,l,c,s,u,f,d,h="bottom",g=0,p=0,m=!1,v=!1,y=!1,x=d3.scaleLinear(),b=.5,S=.05,k=15,w=50,j="transparent",z="d3sm-axis",O="tick-group",A=5,C="black",E=3,L="black",F=2,M=10,Q=10,q=10,B=14,W=8,V=20,D=void 0,K=function(t,e){},_=function(t,e){return String(t).replace("-"," ").replace("_"," ")},P="#333333",R=2,Y=1e3,X=d3.easeExp,T=2,N=!1;return e.label=function(t){return arguments.length?(d=t,e):d},e.tickTickLabelSpacer=function(t){return arguments.length?(Q=t,e):Q},e.tickLabelMargin=function(t){return arguments.length?(q=t,e):q},e.selection=function(n){return arguments.length?(t=n,e):t},e.orient=function(t){return arguments.length?(h=t,e):h},e.spaceX=function(t){return arguments.length?(g=t,e):g},e.spaceY=function(t){return arguments.length?(p=t,e):p},e.overflowQ=function(t){return arguments.length?(m=t,e):m},e.categoricalQ=function(t){return arguments.length?(v=t,e):v},e.guideLinesQ=function(t){return arguments.length?(y=t,e):y},e.grouping=function(t){return arguments.length?(a=t,e):a},e.scale=function(t){return arguments.length?(x=t,e):x},e.domainPadding=function(t){return arguments.length?(b=t,e):b},e.objectSpacer=function(t){return arguments.length?(S=t,e):S},e.minObjectSize=function(t){return arguments.length?(k=t,e):k},e.maxObjectSize=function(t){return arguments.length?(w=t,e):w},e.namespace=function(t){return arguments.length?(z=t,e):z},e.backgroundFill=function(t){return arguments.length?(j=t,e):j},e.objectClass=function(t){return arguments.length?(O=t,e):O},e.tickLabels=function(t){return arguments.length?(o=t,e):o},e.tickValues=function(t){return arguments.length?(i=t,e):i},e.numberOfTicks=function(t){return arguments.length?(A=t,e):A},e.lineStroke=function(t){return arguments.length?(C=t,e):C},e.lineStrokeWidth=function(t){return arguments.length?(E=t,e):E},e.tickStroke=function(t){return arguments.length?(L=t,e):L},e.tickStrokeWidth=function(t){return arguments.length?(F=t,e):F},e.tickLength=function(t){return arguments.length?(M=t,e):M},e.tickLabelFontSize=function(t){return arguments.length?(B=t,e):B},e.tickLabelMinFontSize=function(t){return arguments.length?(W=t,e):W},e.tickLabelMaxFontSize=function(t){return arguments.length?(V=t,e):V},e.tickLabelTextAnchor=function(t){return arguments.length?(l=t,e):l},e.tickLabelRotation=function(t){return arguments.length?(c=t,e):c},e.tickLabelFunc=function(t){return arguments.length?(D=t,e):D},e.tickLabelOnClick=function(t){return arguments.length?(K=t,e):K},e.guidelineSpace=function(t){return arguments.length?(s=t,e):s},e.guideLineStroke=function(t){return arguments.length?(P=t,e):P},e.guideLineStrokeWidth=function(t){return arguments.length?(R=t,e):R},e.transitionDuration=function(t){return arguments.length?(Y=t,e):Y},e.easeFunc=function(t){return arguments.length?(X=t,e):X},e.objectSize=function(t){return arguments.length?(u=t,e):u},e.spacerSize=function(t){return arguments.length?(f=t,e):f},e.roundTo=function(t){return arguments.length?(T=t,e):T},e.reverseScaleQ=function(t){return arguments.length?(N=t,e):N},e.tickLabelOnHoverFunc=function(t){return arguments.length?(_=t,e):_},e}function Z(t){function e(){t.on("mouseover",n),t.on("mousemove",n),t.on("mouseout",function(){d3.selectAll(".d3sm-tooltip").remove()})}function n(t,e){_("d3sm-tooltip");var n=i[t],[l,c]=d3.mouse(d3.select("html").node());R("tooltip","mousemove detected",{key:t,index:e,x:l,y:c}),R("tooltip","current data",n);var s=Q(d3.select("html"),"tooltip","d3sm-tooltip").classed("card",!0).style("max-width","300px").style("background-color","#212529").style("color","white"),u=Q(s,"div","card-body"),f=(Q(u,"h5","card-title").text(void 0==o?t:"function"==typeof o?o(t,n,e):o).style("color","cyan"),Q(Q(u,"table","table").classed("table-dark",!0),"tbody"));(f=(f=f.selectAll("tr")).data(void 0==r?d3.keys(n):r)).exit().remove();var d=f.enter().append("tr").style("max-width","300px");d.append("td").attr("class",function(t,e){return"tooltip-key"}),d.append("td").attr("class",function(t,e,n){return"tooltip-value"}).attr("tooltip-row-index",function(t,e){return e}),_("tooltip-rows"),f.selectAll(".tooltip-key").text(function(t,e){return t}),f.selectAll("tr .tooltip-value").text(function(t,e){R("tooltip","trying to set value",{rowKey:t,rowIndex:e});var e=d3.select(this).attr("tooltip-row-index"),r=n[t];return void 0!=a&&"function"==typeof(r=a[e])&&(r=r(n,t)),"number"==typeof r?b(r,5):r}),P(),P(),l+=15;var h=s.node().getBoundingClientRect();l+h.width>window.innerWidth-window.scrollX&&(l=d3.event.pageX-h.width-15),c+h.height>window.innerHeight-window.scrollY&&(c=d3.event.pageY-h.height-15),"relative"==s.style("position")?s.style("position","absolute").style("left",l+"px").style("top",c+"px"):s.style("left",l+"px").style("top",c+"px"),s.attr("z-index",1e4)}var r,a,o,i,t;return e.keys=function(t){return arguments.length?(r=t,e):r},e.values=function(t){return arguments.length?(a=t,e):a},e.header=function(t){return arguments.length?(o=t,e):o},e.data=function(t){return arguments.length?(i=t,e):i},e.selection=function(n){return arguments.length?(t=n,e):t},e}function I(){function t(t,r,a,s,h){var p,m="fill"==s?l:i;if(e(),"index"==c)p=void 0!=s?o(g(d(a)),m):g(d(a));else if("value"==c){y=u(t,r,a);p=void 0!=s?o(g(d(y)),m):g(d(y))}else if("category"==c){var v=f(t,r,a),y=n.indexOf(v);p=void 0!=s?o(g(d(y)),m):g(d(y))}else p=void 0!=s?o(g(d(a)),m):g(d(a));return p}function e(){h.domain([0,r.length]),"category"==c&&void 0!=n?h.range([0,n.length]):h.range(s);var t=Array(r.length).fill(0).map(function(t,e){return h(e)});d.domain(t)}var n,r=["#2c7bb6","#00a6ca","#00ccbc","#90eb9d","#ffff8c","#f9d057","#f29e2e","#e76818","#d7191c"],a=d3.interpolateRgb,o=y,i=0,l=.4,c="index",s=[0,r.length-1],u=function(t,e,n){return e},f=function(t,e,n){return e.category},d=d3.scaleLinear().interpolate(a).domain(s).range(r),h=d3.scaleLinear(),g=function(t){return"#"+t.match(/\d+/g).map(function(t,e){return(+t<16?"0":"")+(+t).toString(16)}).join("")};return t.colors=function(e){return arguments.length?(r=e,d.range(r),t):r},t.interpolation=function(e){return arguments.length?(a=e,d.interpolate(a).range(r),t):a},t.dataExtent=function(e){return arguments.length?(s=e,d.domain(s).interpolate(d.interpolate()),t):s},t.scale=function(e){return arguments.length?(e=e.domain(d.domain()).interpolate(d.interpolate()).range(d.range()),d=e,t):d},t.modifyOpacity=function(e){return arguments.length?(o=e,t):o},t.strokeOpacity=function(e){return arguments.length?(i=e,t):i},t.fillOpacity=function(e){return arguments.length?(l=e,t):l},t.colorBy=function(e){return arguments.length?(c=e,t):c},t.valueExtractor=function(e){return arguments.length?(u=e,t):u},t.categoryExtractor=function(e){return arguments.length?(f=e,t):f},t.categories=function(e){return arguments.length?(n=e,t):n},t}function J(t){function e(){var e={x:0,y:0,width:r,height:a},C=xt.sel.setupContainer(t,w,e,k);o=d3.keys(n),i=o.map(f),l=o.map(g),c=o.map(v);o.length;var E=[Math.min(...i)-u,Math.max(...i)+u],L=[Math.min(...l)-h,Math.max(...l)+h],F=[Math.min(...c)-m,Math.max(...c)+m];s.domain(E).range([0,r]),d.domain(L).range([a,0]),p.domain(F).range([y,x]);var M=C.selectAll("."+j),Q=(M=M.data(o)).enter().append("circle").attr("class",j).attr("cx",0).attr("cy",a).attr("r",0),q=M.exit();(M=M.merge(Q)).each(function(t,e){var r=d3.select(this),a=n[t],o=i[e],u=l[e],f=c[e],h=S(t,a,e,"fill"),g=S(t,a,e,"stroke");r.transition().duration(z).ease(O).attr("cx",s(o)).attr("cy",d(u)).attr("r",p(f)).attr("fill",h).attr("stroke",g).attr("stroke-width",b),r.on("mouseover",function(t,e){M.style("opacity",.2),r.style("opacity",1),r.transition().duration(z/2).ease(O).attr("stroke-width",2*b).attr("r",1.5*p(f))}),r.node().addEventListener("mouseout",function(){C.selectAll("."+j).style("opacity",1),r.transition().duration(z/2).ease(O).attr("stroke-width",b).attr("r",p(f))})}),q.transition().duration(z).ease(O).attr("cx",0).attr("cy",a).attr("r",0).remove(),A.selection(M).data(n),A()}var n,r,a,o,i,l,c,s=d3.scaleLinear(),u=.5,f=function(t,e){return n[t].x},d=d3.scaleLinear(),h=.5,g=function(t,e){return n[t].y},p=d3.scaleLinear(),m=.5,v=function(t,e){return 2},y=2,x=10,b=2,S=I(),k="transparent",w="d3sm-scatter",j="scatter-point",z=1e3,O=d3.easeExp,A=Z();return e.selection=function(n){return arguments.length?(t=n,e):t},e.data=function(t){return arguments.length?(n=t,e):n},e.spaceX=function(t){return arguments.length?(r=t,e):r},e.spaceY=function(t){return arguments.length?(a=t,e):a},e.scaleX=function(t){return arguments.length?(s=t,e):s},e.domainPaddingX=function(t){return arguments.length?(u=t,e):u},e.valueExtractorX=function(t){return arguments.length?(f=t,e):f},e.scaleY=function(t){return arguments.length?(d=t,e):d},e.domainPaddingY=function(t){return arguments.length?(h=t,e):h},e.valueExtractorY=function(t){return arguments.length?(g=t,e):g},e.scaleR=function(t){return arguments.length?(p=t,e):p},e.domainPaddingR=function(t){return arguments.length?(m=t,e):m},e.valueExtractorR=function(t){return arguments.length?(v=t,e):v},e.minRadius=function(t){return arguments.length?(y=t,e):y},e.maxRadius=function(t){return arguments.length?(x=t,e):x},e.pointStrokeWidth=function(t){return arguments.length?(b=t,e):b},e.colorFunction=function(t){return arguments.length?(S=t,e):S},e.backgroundFill=function(t){return arguments.length?(k=t,e):k},e.namespace=function(t){return arguments.length?(w=t,e):w},e.objectClass=function(t){return arguments.length?(j=t,e):j},e.transitionDuration=function(t){return arguments.length?(z=t,e):z},e.easeFunc=function(t){return arguments.length?(O=t,e):O},e.pointKeys=function(t){return arguments.length?(o=t,e):o},e.valuesX=function(t){return arguments.length?(i=t,e):i},e.valuesY=function(t){return arguments.length?(l=t,e):l},e.valuesR=function(t){return arguments.length?(c=t,e):c},e.tooltip=function(t){return arguments.length?(A=t,e):A},e}function U(t){function e(){var e="horizontal"==u||"bottom"==u||"top"==u,C=!e,E={x:0,y:0,width:r,height:a},L=xt.sel.setupContainer(t,k,E,S);i=d3.keys(n),l=i.map(d);var F=void 0==o?i.sort(h):o,M=(i=xt.arr.flatten(F)).length,Q=[Math.min(...l)-p,Math.max(...l)+p];g.domain(Q).range(e?[0,a]:"right"==u?[0,r]:[r,0]);var q=e?r:a;c=void 0==c?xt.math.calculateWidthOfObject(q,M,v,y,m,f):c,s=void 0==s?xt.math.calculateWidthOfSpacer(i,q,c,M,m,f):s;var B=G().horizontalQ(e).scale(g).moveby("category").numberOfObjects(M).objectClass(w).objectSize(c).spacerSize(s).transitionDuration(j).easeFunc(z).namespace(k),W=B.exitFunction();B.exitFunction(function(t){void 0==c&&console.log(t.nodes(),c),W(t),t.selectAll("g").classed("to-remove",!0),t.selectAll("* > rect").transition().duration(j).attr("transform",function(t,e){return"translate(0,"+(C?0:g(Q[1]))+")"}).attr("width",e?c:0).attr("height",C?c:0).remove()}),B(L,F,0);var V=[];L.selectAll("g:not(.to-remove)."+w).each(function(t,e){V.push(Number(d3.select(this).attr("parent-index")))}),b="index"==b.colorBy()?b.dataExtent([0,Math.max(...V)]):b.dataExtent(Q),L.selectAll("g."+w+":not(.to-remove)").each(function(t,r){var a=d3.select(this),o=(n[t],d(t,r)),r=void 0==a.attr("parent-index")?r:a.attr("parent-index"),i=b(t,o,r,"fill"),l=b(t,o,r,"stroke"),s=xt.sel.safeSelect(a,"rect","bar-rect");void 0==s.attr("transform")&&s.attr("transform",function(t,e){return"translate(0,"+(C?0:g(Q[1]))+")"}).attr("width",e?c:0).attr("height",C?c:0),s.transition().duration(j).ease(z).attr("transform",function(t,n){return"translate("+(e?c-c*A:"right"==u?g(Q[1])-g(o):c-c*A)+","+(C?c-c*A:g(Q[1])-g(o))+")"}).attr("width",e?c*A:g(o)).attr("height",C?c*A:g(o)).attr("fill",i).attr("stroke",l).attr("stroke-width",x),a.on("mouseover",function(t,e){L.selectAll("g."+w).style("opacity",.2),a.style("opacity",1),s.attr("stroke-width",2*x)}),a.on("mouseout",function(){L.selectAll("g."+w).style("opacity",1),s.attr("stroke-width",x)})}),O.selection(L.selectAll(".bar-rect")).data(n),O()}var n,r,a,o,i,l,c,s,u="horizontal",f=!1,d=function(t,e){return n[t]},h=function(t,e){return d3.descending(n[t],n[e])},g=d3.scaleLinear(),p=.5,m=.05,v=50,y=100,x=2,b=I(),S="transparent",k="d3sm-bar",w="bar",j=1e3,z=d3.easeExp,O=Z(),A=1;return e.selection=function(n){return arguments.length?(t=n,e):t},e.data=function(t){return arguments.length?(n=t,e):n},e.orient=function(t){return arguments.length?(u=t,e):u},e.spaceX=function(t){return arguments.length?(r=t,e):r},e.spaceY=function(t){return arguments.length?(a=t,e):a},e.overflowQ=function(t){return arguments.length?(f=t,e):f},e.grouping=function(t){return arguments.length?(o=t,e):o},e.valueExtractor=function(t){return arguments.length?(d=t,e):d},e.sortingFunction=function(t){return arguments.length?(h=t,e):h},e.scale=function(t){return arguments.length?(g=t,e):g},e.domainPadding=function(t){return arguments.length?(p=t,e):p},e.objectSpacer=function(t){return arguments.length?(m=t,e):m},e.minObjectSize=function(t){return arguments.length?(v=t,e):v},e.maxObjectSize=function(t){return arguments.length?(y=t,e):y},e.barStrokeWidth=function(t){return arguments.length?(x=t,e):x},e.colorFunction=function(t){return arguments.length?(b=t,e):b},e.backgroundFill=function(t){return arguments.length?(S=t,e):S},e.namespace=function(t){return arguments.length?(k=t,e):k},e.objectClass=function(t){return arguments.length?(w=t,e):w},e.transitionDuration=function(t){return arguments.length?(j=t,e):j},e.easeFunc=function(t){return arguments.length?(z=t,e):z},e.barKeys=function(t){return arguments.length?(i=t,e):i},e.barValues=function(t){return arguments.length?(l=t,e):l},e.objectSize=function(t){return arguments.length?(c=t,e):c},e.spacerSize=function(t){return arguments.length?(s=t,e):s},e.tooltip=function(t){return arguments.length?(O=t,e):O},e.barPercent=function(t){return arguments.length?(A=t,e):A},e}function $(t){function e(){var e={x:0,y:0,width:r,height:a},g=xt.sel.setupContainer(t,L,e,E);(o=d3.keys(n)).sort(function(t,e){return q(t,e)||B(t,e)}),xt.con.log("bubbleHeatmap","cells are sorted by",o),i=xt.arr.unique(o.map(y)),l=xt.arr.unique(o.map(x)),c=xt.arr.unique(o.map(b)),s=xt.arr.unique(o.map(S)),xt.con.log("bubbleHeatmap","x and y keys are",{x:i,y:l});var p=i.length,m=l.length,v=[Math.min(...c)-j,Math.max(...c)+j];d=xt.math.calculateWidthOfObject(a,m,O,A,z,k),u=xt.math.calculateWidthOfObject(r,p,O,A,z,k),h=xt.math.calculateWidthOfSpacer(l,a,d,m,z,k),f=xt.math.calculateWidthOfSpacer(i,r,u,p,z,k),xt.con.log("bubbleHeatmap","size of",{x:u,y:d}),w.domain(v).range([Math.min(O/2,Math.min(d,u)/2),Math.min(d,u)/2]);var D=G().horizontalQ(!1).moveby("category").numberOfObjects(m).objectClass(xt.str.hypenate(F,"row")).objectSize(d).spacerSize(h).transitionDuration(M).easeFunc(Q).namespace("row"),K=G().horizontalQ(!0).moveby("category").numberOfObjects(p).objectClass(F).objectSize(u).spacerSize(f).transitionDuration(M).easeFunc(Q);D(g,l,0),g.selectAll("g."+xt.str.hypenate(F,"row")).each(function(t,e){K(d3.select(this),i,0)});var _=g.selectAll("g:not(.to-remove)."+F).data(o),P=[];_.each(function(t,e){P.push(Number(d3.select(this).attr("parent-index")))}),W="index"==W.colorBy()?W.dataExtent([0,Math.max(...P)]):W.dataExtent([0,Math.max(...s)]),_.each(function(t,e){xt.con.log("bubbleHeatmap","each cell",{key:t,index:e,node:d3.select(this).node()});var r=d3.select(this),a=(n[t],S(t,e)),o=b(t,e),e=void 0==r.attr("parent-index")?e:r.attr("parent-index"),i=W(t,a,e,"fill"),l=W(t,a,e,"stroke");xt.con.log("bubbleHeatmap","radius",{radius:o,scaled:w(o),extent:v,range:w.range()}),xt.sel.safeSelect(r,"circle",xt.str.hypenate(F,"circle")).attr("cx",u/2).attr("cy",d/2).attr("r",w(o)).attr("fill",i).attr("stroke",l).attr("stroke-width",C)}),V.selection(_.selectAll("circle."+xt.str.hypenate(F,"circle"))).data(n),V()}var n,r,a,o,i,l,c,s,u,f,d,h,g="x",p="y",m="r",v="v",y=function(t,e){return n[t][g]},x=function(t,e){return n[t][p]},b=function(t,e){return n[t][m]},S=function(t,e){return n[t][v]},k=!1,w=d3.scaleLinear(),j=.5,z=0,O=50,A=100,C=2,E="transparent",L="d3sm-bubble",F="bubble",M=1e3,Q=d3.easeExp,q=function(t,e){return y(t)-y(e)},B=function(t,e){return x(t)-x(e)},W=I().colorBy("value"),V=Z();return e.selection=function(n){return arguments.length?(t=n,e):t},e.data=function(t){return arguments.length?(n=t,e):n},e.spaceX=function(t){return arguments.length?(r=t,e):r},e.spaceY=function(t){return arguments.length?(a=t,e):a},e.xKey=function(t){return arguments.length?(g=t,e):g},e.yKey=function(t){return arguments.length?(p=t,e):p},e.rKey=function(t){return arguments.length?(m=t,e):m},e.vKey=function(t){return arguments.length?(v=t,e):v},e.cellKeys=function(t){return arguments.length?(o=t,e):o},e.xValues=function(t){return arguments.length?(i=t,e):i},e.yValues=function(t){return arguments.length?(l=t,e):l},e.rValues=function(t){return arguments.length?(c=t,e):c},e.vValues=function(t){return arguments.length?(s=t,e):s},e.xExtractor=function(t){return arguments.length?(y=t,e):y},e.yExtractor=function(t){return arguments.length?(x=t,e):x},e.rExtractor=function(t){return arguments.length?(b=t,e):b},e.vExtractor=function(t){return arguments.length?(S=t,e):S},e.overflowQ=function(t){return arguments.length?(k=t,e):k},e.scale=function(t){return arguments.length?(w=t,e):w},e.domainPadding=function(t){return arguments.length?(j=t,e):j},e.objectSpacer=function(t){return arguments.length?z=t:n},e.minObjectSize=function(t){return arguments.length?(O=t,e):O},e.maxObjectSize=function(t){return arguments.length?(A=t,e):A},e.bubbleStrokeWidth=function(t){return arguments.length?(C=t,e):C},e.backgroundFill=function(t){return arguments.length?(E=t,e):E},e.namespace=function(t){return arguments.length?(L=t,e):L},e.objectClass=function(t){return arguments.length?(F=t,e):F},e.transitionDuration=function(t){return arguments.length?(M=t,e):M},e.easeFunc=function(t){return arguments.length?(Q=t,e):Q},e.tooltip=function(t){return arguments.length?(V=t,e):V},e.colorFunction=function(t){return arguments.length?(W=t,e):W},e.xSize=function(t){return arguments.length?(u=t,e):u},e.xSpacerSize=function(t){return arguments.length?(f=t,e):f},e.ySize=function(t){return arguments.length?(d=t,e):d},e.ySpacerSize=function(t){return arguments.length?(h=t,e):h},e}function tt(t){function e(){var e="horizontal"==u,d=!e,F={x:0,y:0,width:r,height:a},M=xt.sel.setupContainer(t,O,F,z),Q=void 0==o?d3.keys(n).sort(p):o;i=xt.arr.flatten(Q),l=i.map(g);var q=i.length,B=[Math.min(...l.map(function(t,e){return t[h[0]]}))-v,Math.max(...l.map(function(t,e){return t[h[4]]}))+v];m.domain(B).range(e?[0,a]:[r,0]);var W=e?r:a;c=xt.math.calculateWidthOfObject(W,q,x,b,y,f),s=xt.math.calculateWidthOfSpacer(i,W,c,q,y,f),G().horizontalQ(e).scale(m).moveby("category").numberOfObjects(q).objectClass(A).objectSize(c).spacerSize(s).transitionDuration(C).easeFunc(E).namespace(O)(M,Q,0);var V=[];M.selectAll("g:not(.to-remove)."+A).each(function(t,e){xt.arr.hasQ(i,t)&&V.push(Number(d3.select(this).attr("parent-index")))}),k="index"==k.colorBy()?k.dataExtent([0,Math.max(...V)]):k.dataExtent(B),M.selectAll("g:not(.to-remove)."+A).each(function(t,r){var a=d3.select(this),o=(n[t],g(t,r)),i=o[h[0]],l=o[h[1]],s=o[h[2]],f=o[h[3]],p=o[h[4]],r=void 0==a.attr("parent-index")?r:a.attr("parent-index"),v=k(t,s,r,"fill"),y=k(t,s,r,"stroke"),x=xt.sel.safeSelect(a,"g","whisker"),b=xt.sel.safeSelect(x,"path","upper"),z=xt.sel.safeSelect(x,"path","lower"),O=xt.sel.safeSelect(a,"g","quartile"),A=xt.sel.safeSelect(O,"rect","upper"),L=xt.sel.safeSelect(O,"rect","lower"),F=xt.sel.safeSelect(O,"circle","median");A.transition().duration(C).ease(E).attr("width",e?c:m(f)-m(s)).attr("height",d?c:m(f)-m(s)).attr("fill",v).attr("stroke",y).attr("stroke-width",w).attr("transform",function(t,n){return"translate("+(e?0:m(s))+","+(d?0:m(B[1])-m(f))+")"}),L.transition().duration(C).ease(E).attr("width",e?c:m(s)-m(l)).attr("height",d?c:m(s)-m(l)).attr("fill",v).attr("stroke",y).attr("stroke-width",w).attr("transform",function(t,n){return"translate("+(e?0:m(l))+","+(d?0:m(B[1])-m(s))+")"}),F.transition().duration(C).ease(E).attr("r",function(t,e){var n=c/2,r=(m(f)-m(l))/2;return n>r?r:n}).attr("fill",v).attr("stroke",y).attr("stroke-width",w).attr("transform",function(t,n){return"translate("+(e?c/2:m(s))+","+(d?c/2:m(B[1])-m(s))+")"}),z.transition().duration(C).ease(E).attr("d",function(t,n){var r=e?m(l)-m(i):c,a=d?m(l)-m(i):c;return xt.paths.whiskerPath(!1,0,0,a,r,S,u)}).attr("transform",function(t,n){return"translate("+(e?0:m(l))+","+(d?0:m(B[1])-m(l))+")"}).attr("stroke","black").attr("stroke-width",j).attr("fill","none"),b.transition().duration(C).ease(E).attr("d",function(t,n){var r=e?m(p)-m(f):c,a=d?m(p)-m(f):c;return xt.paths.whiskerPath(!0,0,0,a,r,S,u)}).attr("transform",function(t,n){return"translate("+(e?0:m(f))+","+(d?0:m(B[1])-m(p))+")"}).attr("stroke","black").attr("stroke-width",j).attr("fill","none")}),L.selection(M.selectAll("g:not(.to-remove)."+A)).data(n),L()}var n,r,a,o,i,l,c,s,u="horizontal",f=!0,d="quartiles",h=["0.00","0.25","0.50","0.75","1.00"],g=function(t,e){return n[t][d]},p=function(t,e){return d3.descending(g(t)[h[4]],g(e)[h[4]])},m=d3.scaleLinear(),v=.5,y=.05,x=15,b=50,S=.6,k=I(),w=2,j=2,z="transparent",O="d3sm-box-whisker",A="box-whisk",C=1e3,E=d3.easeExp,L=Z();return e.selection=function(n){return arguments.length?(t=n,e):t},e.data=function(t){return arguments.length?(n=t,e):n},e.orient=function(t){return arguments.length?(u=t,e):u},e.spaceX=function(t){return arguments.length?(r=t,e):r},e.spaceY=function(t){return arguments.length?(a=t,e):a},e.overflowQ=function(t){return arguments.length?(f=t,e):f},e.grouping=function(t){return arguments.length?(o=t,e):o},e.quartilesKey=function(t){return arguments.length?(d=t,e):d},e.quartilesKeys=function(t){return arguments.length?(h=t,e):h},e.valueExtractor=function(t){return arguments.length?(g=t,e):g},e.sortingFunction=function(t){return arguments.length?(p=t,e):p},e.scale=function(t){return arguments.length?(m=t,e):m},e.domainPadding=function(t){return arguments.length?(v=t,e):v},e.objectSpacer=function(t){return arguments.length?(y=t,e):y},e.minObjectSize=function(t){return arguments.length?(x=t,e):x},e.maxObjectSize=function(t){return arguments.length?(b=t,e):b},e.whiskerWidthPercent=function(t){return arguments.length?(S=t,e):S},e.colorFunction=function(t){return arguments.length?(k=t,e):k},e.boxStrokeWidth=function(t){return arguments.length?(w=t,e):w},e.whiskerStrokeWidth=function(t){return arguments.length?(j=t,e):j},e.backgroundFill=function(t){return arguments.length?(z=t,e):z},e.namespace=function(t){return arguments.length?(O=t,e):O},e.objectClass=function(t){return arguments.length?(A=t,e):A},e.transitionDuration=function(t){return arguments.length?(C=t,e):C},e.easeFunc=function(t){return arguments.length?(E=t,e):E},e.boxKeys=function(t){return arguments.length?(i=t,e):i},e.boxValues=function(t){return arguments.length?(l=t,e):l},e.objectSize=function(t){return arguments.length?(c=t,e):c},e.spacerSize=function(t){return arguments.length?(s=t,e):s},e.tooltip=function(t){return arguments.length?(L=t,e):L},e}function et(t){function e(){var e={x:0,y:0,width:r,height:a},h=xt.sel.setupContainer(t,A,e,O);o=d3.keys(n),i=xt.arr.unique(o.map(m)),l=xt.arr.unique(o.map(v)),c=xt.arr.unique(o.map(y)),o.sort(function(t,e){return F(t,e)||M(t,e)}),xt.con.log("heatmap","cells are sorted by",o),xt.con.log("heatmap","x and y keys are",{x:i,y:l});var g=i.length,p=l.length;f=xt.math.calculateWidthOfObject(a,p,S,j,b,x),s=xt.math.calculateWidthOfObject(r,g,k,w,b,x),d=xt.math.calculateWidthOfSpacer(l,a,f,p,b,x),u=xt.math.calculateWidthOfSpacer(i,r,s,g,b,x),xt.con.log("heatmap","size of",{x:s,y:f});var B=G().horizontalQ(!1).moveby("category").numberOfObjects(p).objectClass(xt.str.hypenate(C,"row")).objectSize(f+d).spacerSize(0).transitionDuration(E).easeFunc(L).namespace("row"),W=G().horizontalQ(!0).moveby("category").numberOfObjects(g).objectClass(C).objectSize(s+u).spacerSize(0).transitionDuration(E).easeFunc(L);B(h,l,0),h.selectAll("g."+xt.str.hypenate(C,"row")).each(function(t,e){W(d3.select(this),i,0)});var V=h.selectAll("g:not(.to-remove)."+C);if(o.length!=l.length*i.length){var D={};o.map(function(t,e){D[m(t)+"::"+v(t)]=t});for(var K=[],_=0;_.5?l:-l}).attr("cx",function(t,n){var r=o.pointValues[n];if(e){var a=xt.arr.whichBin(o.binned,r),i=Math.random(),l=T(i*o.frequencies[a]*.5);return Math.random()>.5?l:-l}return p(r)}).attr("stroke",function(t,e){var t=o.pointValues[e];return k(t,"stroke",s,z,O)}).attr("fill",function(t,e){var t=o.pointValues[e];return k(t,"fill",s,z,O)}).attr("stroke-width",j),v.selectAll("circle.point").on("mouseover",function(t,e){B.selectAll("g."+A).style("opacity",.2),a.style("opacity",1),f.attr("stroke-width",2*b),h.attr("stroke-width",2*b),B.selectAll(".point").style("opacity",.2),d3.select(this).style("opacity",1).attr("r",2*w).attr("stroke-width",2*j)}),v.selectAll("circle.point").on("mouseout",function(t,e){var n=document.createEvent("SVGEvents");n.initEvent("mouseout",!0,!0),u.node().dispatchEvent(n),B.selectAll(".point").style("opacity",1),d3.select(this).attr("stroke-width",j).attr("r",w)})}else cV.selectAll(".point").transition().duration(C).ease(E).attr("r",0).attr("cy",e?p(P[1])-p(m):T(0)).attr("cx",e?T(0):p(m)).remove()}}),F.selection(B.selectAll("g:not(.to-remove)."+A+" .area")),void 0==F.data()&&F.data(n),F(),void 0==F.values()&&F.values([function(t,e){return t.quartiles[e]},function(t,e){return t.quartiles[e]},function(t,e){return t.quartiles[e]},function(t,e){return t.quartiles[e]},function(t,e){return t.quartiles[e]}])}var n,r,a,o,i,l,c,s,u="horizontal",f=!0,d=!0,h=function(t,e){return n[t]},g=function(t,e){return d3.descending(n[t],n[e])},p=d3.scaleLinear(),m=.5,v=.05,y=50,x=100,b=2,S=I(),k=function(t,e,n,r,a){var o=d3.scaleLinear().domain([r,a]).range([-.25,.05]),i=xt.color.modifyHexidecimalColorLuminance(n.replace("#",""),o(t)),l="stroke"==e?0:.25;return xt.color.modifyHexidecimalColorLuminance(i.replace("#",""),l)},w=3,j=2,z="transparent",O="d3sm-violin",A="violin",C=1e3,E=d3.easeExp,L=["Q0","Q1","Q2","Q3","Q4"],F=Z().keys([L[4],L[3],L[2],L[1],L[0]]),M=Z(),Q=function(t,e){return e.points},q=function(t,e){return e[t].value};return e.violinPointsExtractor=function(t){return arguments.length?(Q=t,e):Q},e.violinPointValueExtractor=function(t){return arguments.length?(q=t,e):q},e.selection=function(n){return arguments.length?(t=n,e):t},e.data=function(t){return arguments.length?(n=t,e):n},e.orient=function(t){return arguments.length?(u=t,e):u},e.spaceX=function(t){return arguments.length?(r=t,e):r},e.spaceY=function(t){return arguments.length?(a=t,e):a},e.overflowQ=function(t){return arguments.length?(f=t,e):f},e.pointsQ=function(t){return arguments.length?(d=t,e):d},e.grouping=function(t){return arguments.length?(o=t,e):o},e.valueExtractor=function(t){return arguments.length?(h=t,e):h},e.sortingFunction=function(t){return arguments.length?(g=t,e):g},e.scale=function(t){return arguments.length?(p=t,e):p},e.domainPadding=function(t){return arguments.length?(m=t,e):m},e.objectSpacer=function(t){return arguments.length?(v=t,e):v},e.minObjectSize=function(t){return arguments.length?(y=t,e):y},e.maxObjectSize=function(t){return arguments.length?(x=t,e):x},e.objectStrokeWidth=function(t){return arguments.length?(b=t,e):b},e.colorFunction=function(t){return arguments.length?(S=t,e):S},e.pointColorFunc=function(t){return arguments.length?(k=t,e):k},e.pointRadius=function(t){return arguments.length?(w=t,e):w},e.pointStrokeWidth=function(t){return arguments.length?(j=t,e):j},e.backgroundFill=function(t){return arguments.length?(z=t,e):z},e.namespace=function(t){return arguments.length?(O=t,e):O},e.objectClass=function(t){return arguments.length?(A=t,e):A},e.transitionDuration=function(t){return arguments.length?(C=t,e):C},e.easeFunc=function(t){return arguments.length?(E=t,e):E},e.quartileKey=function(t){return arguments.length?(quartileKey=t,e):quartileKey},e.quartileKeys=function(t){return arguments.length?(L=t,e):L},e.violinKeys=function(t){return arguments.length?(i=t,e):i},e.violinValues=function(t){return arguments.length?(l=t,e):l},e.objectSize=function(t){return arguments.length?(c=t,e):c},e.spacerSize=function(t){return arguments.length?(s=t,e):s},e.tooltip=function(t){return arguments.length?(F=t,e):F},e.pointsTooltip=function(t){return arguments.length?(M=t,e):M},e}function rt(){function t(t,o){var i=o[t],l=e(t,i),c=d3.keys(l),s=c.map(function(t,e){return n(t,l)}),u=xt.math.quartiles(s,a),f=d3.histogram()(s),d=f.map(t=>t.length),h=r?{x:0,y:d3.min(s)}:{x:d3.min(s),y:0},g=r?{x:0,y:d3.max(s)}:{x:d3.max(s),y:0},p=f.map(function(t,e){return r?{y:t.length?d3.median(t):d3.median([t.x0,t.x1]),x:d[e]}:{x:t.length?d3.median(t):d3.median([t.x0,t.x1]),y:d[e]}});return p=[h].concat(p).concat([g]),i.binned=f,i.frequencies=d,i.contour=p,i.quartiles=u,i.pointKeys=c,i.pointValues=s,i}var e,n,r=!0,a=["Q0","Q1","Q2","Q3","Q4"];return t.horizontalQ=function(e){return arguments.length?(r=e,t):r},t.quartileKeys=function(e){return arguments.length?(a=e,t):a},t.violinPointsExtractor=function(n){return arguments.length?(e=n,t):e},t.violinPointValueExtractor=function(e){return arguments.length?(n=e,t):n},t}function at(t){function e(){var e="horizontal"==p,n=!e,r={x:0,y:0,width:o,height:i},x=xt.sel.setupContainer(t,S,r,b);l=d3.keys(a),c=xt.arr.unique(l.map(z)).sort(),s=xt.arr.unique(l.map(O)).sort().sort(function(t,e){return t.split(";").length-e.split(";").length}),e?l.sort(function(t,e){return F(t,e)||L(t,e)}):l.sort(function(t,e){return L(t,e)||F(t,e)});var A=e?s:c,M=e?c:s,Q=e?A.length:M.length,q=e?M.length:A.length;h=xt.math.calculateWidthOfObject(o,Q,v,y,C,m),f=xt.math.calculateWidthOfObject(i,q,v,y,E,m),g=xt.math.calculateWidthOfSpacer(A,o,h,Q,C,m),d=xt.math.calculateWidthOfSpacer(M,i,f,q,E,m);var B=G().horizontalQ(!1).moveby("category").numberOfObjects(q).objectSize(f).spacerSize(d).transitionDuration(w).easeFunc(j),W=G().horizontalQ(!0).moveby("category").numberOfObjects(Q).objectClass(k).objectSize(h).spacerSize(g).transitionDuration(w).easeFunc(j);n?(W.objectClass(k),B.namespace("across").objectClass(xt.str.hypenate(k,"across")),B(x,M,0),x.selectAll("g."+xt.str.hypenate(k,"across")).each(function(t,e){W(d3.select(this),A,0)})):(W.namespace("across").objectClass(xt.str.hypenate(k,"across")),B.objectClass(k),W(x,A,0),x.selectAll("g."+xt.str.hypenate(k,"across")).each(function(t,e){B(d3.select(this),M,0)}));var V=x.selectAll("g:not(.to-remove)."+k),D={};l.map(function(t,e){D[z(t)+"::"+O(t)]=t}),V.data(l),V.each(function(t,e){var n=d3.select(this);if(void 0!=t){a[t];var r=z(t,e),o=O(t,e);n.classed(o,!0),n.classed(r,!0),xt.sel.safeSelect(n,"circle",xt.str.hypenate(k,"circle")).attr("cx",h/2).attr("cy",f/2).attr("r",void 0==u?Math.min(h,f)/2:u).attr("fill",o.includes(r)?"black":"rgb(233,233,233)").attr("stroke","black").attr("in-intersection",o.includes(r))}})}function n(){var t={};return s.map(function(e,n){t[e]={total:0}}),l.map(function(e,n){var r=A(e,n);0==t[O(e,n)].total&&(Array.isArray(r)?(t[O(e,n)].total+=r.length,t[O(e,n)].values=r):t[O(e,n)].total+=r)}),t}function r(){var t={};return c.map(function(e,n){t[e]={total:0}}),l.map(function(e,n){var r=A(e,n);Array.isArray(r)?t[z(e,n)].total+=r.length:t[z(e,n)].total+=r}),t}var a,o,i,l,c,s,u,f,d,h,g,p="horizontal",m=!1,v=20,y=50,x=2,b="transparent",S="d3sm-upset",k="upset",w=1e3,j=d3.easeExp,z=function(t,e){return a[t].set},O=function(t,e){return a[t].intersection},A=function(t,e){return a[t].elements},C=.05,E=.05,L=function(t,e){return c.indexOf(z(t))-c.indexOf(z(e))},F=function(t,e){return s.indexOf(O(t))-s.indexOf(O(e))};return e.selection=function(n){return arguments.length?(t=n,e):t},e.data=function(t){return arguments.length?(a=t,e):a},e.orient=function(t){return arguments.length?(p=t,e):p},e.spaceX=function(t){return arguments.length?(o=t,e):o},e.spaceY=function(t){return arguments.length?(i=t,e):i},e.overflowQ=function(t){return arguments.length?(m=t,e):m},e.minObjectSize=function(t){return arguments.length?(v=t,e):v},e.maxObjectSize=function(t){return arguments.length?(y=t,e):y},e.circleStrokeWidth=function(t){return arguments.length?(x=t,e):x},e.backgroundFill=function(t){return arguments.length?(b=t,e):b},e.namespace=function(t){return arguments.length?(S=t,e):S},e.objectClass=function(t){return arguments.length?(k=t,e):k},e.transitionDuration=function(t){return arguments.length?(w=t,e):w},e.easeFunc=function(t){return arguments.length?(j=t,e):j},e.cellKeys=function(t){return arguments.length?(l=t,e):l},e.setValues=function(t){return arguments.length?(c=t,e):c},e.intersectionValues=function(t){return arguments.length?(s=t,e):s},e.xObjectSpacer=function(t){return arguments.length?(C=t,e):C},e.yObjectSpacer=function(t){return arguments.length?(E=t,e):E},e.radius=function(t){return arguments.length?(u=t,e):u},e.setExtractor=function(t){return arguments.length?(z=t,e):z},e.intersectionExtractor=function(t){return arguments.length?(O=t,e):O},e.elementExtractor=function(t){return arguments.length?(A=t,e):A},e.setKeySortingFunction=function(t){return arguments.length?(L=t,e):L},e.intersectionKeySortingFunction=function(t){return arguments.length?(F=t,e):F},e.yObjectSize=function(t){return arguments.length?(f=t,e):f},e.ySpacerSize=function(t){return arguments.length?(d=t,e):d},e.xObjectSize=function(t){return arguments.length?(h=t,e):h},e.xSpacerSize=function(t){return arguments.length?(g=t,e):g},e.intersectionTotals=n,e.setTotals=r,e}function ot(t){function e(){var e="horizontal"==l,r=!e,s={x:0,y:0,width:a,height:o},S=xt.sel.setupContainer(t,v,s,m);p.dataExtent([0,n.length-1]).colorBy("categories").categoryExtractor(function(t,e,n){return e});var k=Math.min(a,o)/2,w=n.length,j=void 0==i?n.sort(u):i,z=xt.arr.flatten(j),O=e?a:o,A=xt.math.calculateWidthOfObject(O,w,d,h,f,c),C=xt.math.calculateWidthOfSpacer(z,O,A,w,f,c);G().horizontalQ(e).scale(scale).moveby("category").numberOfObjects(w).objectClass(y).objectSize(A).spacerSize(C).transitionDuration(x).easeFunc(b).namespace(v)(S,j,0);k=Math.min(A,a,o)/2-g;S.selectAll("g:not(.to-remove)."+y).each(function(t,n){var o=d3.select(this),i=xt.sel.safeSelect(o,"circle"),l=p(void 0,t,n,"fill"),c=p(void 0,t,n,"stroke"),s=e?k+g:(a-2*k)/2+k,u=r?k+g:(a-2*k)/2+k;i.attr("r",k).attr("cx",s).attr("cy",u).attr("fill",l).attr("stroke",c).attr("stroke-width",g);var f=xt.sel.safeSelect(o,"text");f.text(t).attr("text-anchor","middle").attr("transform",function(t,e){return"translate("+s+","+(u+f.node().getBoundingClientRect().height/4)+")"})})}var n,r,a,o,i,l="horizontal",c=!1,s=function(t,e){return r[t]},u=function(t,e){return d3.ascending(t,e)},f=.05,d=10,h=100,g=2,p=I(),m="transparent",v="d3sm-legend",y="legend",x=1e3,b=d3.easeExp;return e.categories=function(t){return arguments.length?(n=t,e):n},e.selection=function(n){return arguments.length?(t=n,e):t},e.data=function(t){return arguments.length?(r=t,e):r},e.orient=function(t){return arguments.length?(l=t,e):l},e.spaceX=function(t){return arguments.length?(a=t,e):a},e.spaceY=function(t){return arguments.length?(o=t,e):o},e.overflowQ=function(t){return arguments.length?(c=t,e):c},e.grouping=function(t){return arguments.length?(i=t,e):i},e.valueExtractor=function(t){return arguments.length?(s=t,e):s},e.sortingFunction=function(t){return arguments.length?(u=t,e):u},e.objectSpacer=function(t){return arguments.length?(f=t,e):f},e.minObjectSize=function(t){return arguments.length?(d=t,e):d},e.maxObjectSize=function(t){return arguments.length?(h=t,e):h},e.bubbleStrokeWidth=function(t){return arguments.length?(g=t,e):g},e.colorFunction=function(t){return arguments.length?(p=t,e):p},e.backgroundFill=function(t){return arguments.length?(m=t,e):m},e.namespace=function(t){return arguments.length?(v=t,e):v},e.objectClass=function(t){return arguments.length?(y=t,e):y},e.transitionDuration=function(t){return arguments.length?(x=t,e):x},e.easeFunc=function(t){return arguments.length?(b=t,e):b},e}function it(t){function e(){var e={x:0,y:0,width:r,height:a},f=xt.sel.setupContainer(t,c,e,u),d=xt.sel.safeSelect(t,"defs"),h=xt.sel.safeSelect(d,"linearGradient").attr("x1","0%").attr("y1","100%").attr("x2","0%").attr("y2","0%").attr("id",xt.str.hypenate(c,"numerical-legend-gradient"));l.dataExtent([o,i]).colorBy("value").valueExtractor(function(t,e,n){return e}),h.selectAll("stop").data(l.colors()).enter().append("stop").attr("offset",function(t,e){return e/(l.colors().length-1)}).attr("stop-color",function(t){return t});var g=xt.sel.safeSelect(f,"rect","legend").attr("transform","translate(0,"+s+")").style("fill","url(#"+xt.str.hypenate(c,"numerical-legend-gradient")+")").attr("x",0).attr("y",0).attr("width",r).attr("height",a-2*s).on("mousemove",function(t,e){n(t,e,g,d3.select(this))}).on("mouseout",function(t,e){d3.select("#"+xt.str.hypenate(c,"legend-tooltip")).remove()});xt.sel.safeSelect(f,"text","min").text(xt.math.round(o,2)).attr("text-anchor","middle").attr("font-size",s+"px").attr("transform",function(t,e){return"translate("+r/2+","+(a-s/4)+")"}),xt.sel.safeSelect(f,"text","max").text(xt.math.round(i,2)).attr("text-anchor","middle").attr("font-size",s+"px").attr("transform",function(t,e){return"translate("+r/2+","+s+")"})}function n(t,e,n,r){var a=d3.scaleLinear().domain([0,n.attr("height")]).range([i,o]),u=d3.mouse(n.node()),h=xt.math.round(a(u[1]),d),g=l(void 0,h,void 0,"stroke"),p=l(void 0,h,void 0,"fill"),m=xt.sel.safeSelect(d3.select("body"),"div",xt.str.hypenate(c,"legend-tooltip")).attr("id",xt.str.hypenate(c,"legend-tooltip")).style("position","absolute").style("left",d3.event.pageX+15+"px").style("top",d3.event.pageY+15+"px").style("background-color",p).style("border-color",g).style("min-width",s*(String(i).split(".")[0].length+3)+"px").style("min-height",s*(String(i).split(".")[0].length+3)+"px").style("border-radius","50%").style("border-radius","5000px").style("display","flex").style("justify-content","center").style("text-align","middle").style("padding","2px").style("border-style","solid").style("border-width",2);xt.sel.safeSelect(m,"div").text(h).style("color",f).style("align-self","center")}var r,a,o=0,i=1,l=I(),c="d3sm-linear-vertical-gradient",s=12,u="transparent",f="black",d=2;return e.min=function(t){return arguments.length?(o=t,e):o},e.max=function(t){return arguments.length?(i=t,e):i},e.spaceX=function(t){return arguments.length?(r=t,e):r},e.spaceY=function(t){return arguments.length?(a=t,e):a},e.namespace=function(t){return arguments.length?(c=t,e):c},e.fontSize=function(t){return arguments.length?(s=t,e):s},e.backgroundFill=function(t){return arguments.length?(u=t,e):u},e.colorFunction=function(t){return arguments.length?(l=t,e):l},e.textColor=function(t){return arguments.length?(f=t,e):f},e.roundTo=function(t){return arguments.length?(d=t,e):d},e}function lt(t){var e=t.attr("transform"),[n,r]=e.split("translate("),[a,o]=r.split(",");return o.split(")"),[parseFloat(a),parseFloat(o)]}function ct(t){function e(){z&&i()}function n(t){z=void 0!=t?t:!z,y=lt(v),x=lt(p),z?g.node().addEventListener("mousedown",o,!0):(g.node().removeEventListener("mousedown",o,!0),a())}function r(){y=lt(v),x=lt(p);var t=xt.sel.safeSelect(p,"g","lasso-container").selectAll('path[instance="'+E+'"]'),e=((t=t.data(A)).exit().remove(),t.enter().append("path"));l(t=t.merge(e))}function a(){var t=xt.sel.safeSelect(p,"g","lasso-container");t.selectAll('path[instance="'+E+'"]').remove();t.remove(),p.selectAll(m).classed("in-lasso",!1),f()}function o(t){t.preventDefault(),t.stopPropagation();var e=xt.sel.safeSelect(p,"g","lasso-container");O=[],g.node().addEventListener("mousemove",c),g.node().addEventListener("mouseup",function(t){g.node().removeEventListener("mousemove",c),A.push(O),A=xt.arr.unique(A)}),l(w=e.append("path").data([O]))}function i(){var t=xt.sel.safeSelect(p,"g","lasso-container").selectAll('path[instance="'+E+'"]'),e=((t=t.data(A)).exit().remove(),t.enter().append("path"));l(t=t.merge(e).transition().duration(_).ease(P))}function l(t){t.attr("class",xt.str.hypenate(j,"lasso-path")).style("opacity",Q).attr("fill",F).attr("d",C).attr("instance",E).style("stroke-dasharray",q).attr("stroke",B).attr("stroke-width",W).style("animation","lassoDash "+M+" linear").style("animation-iteration-count","infinite")}function c(t){if(void 0!=b&&b.dispatch(xt.str.hypenate(j,"drag")),1==t.which){d3.event=t;var e=d3.mouse(p.node()),e=d3.mouse(g.node());if(void 0!=S&&(e[0]=S.invert(e[0])),void 0!=k&&(e[1]=k.invert(e[1])),e[0]=e[0]-y[0]-x[0],e[1]=e[1]-y[1]-x[1],O.length){var n=O[O.length-1],r=[e[0],e[1]],a=[n[0],n[1]];S&&(a[0]=S(a[0]),r[0]=S(r[0])),k&&(a[1]=k(a[1]),r[1]=k(r[1])),xt.math.euclideanDistance(a,r)>L&&s(e)}else s(e)}}function s(t){if(void 0!=t){if(O.push(t),w.attr("d",C),O.length<3)return;u(A.concat([O]))}else u(A)}function u(t){return void 0==t&&(t=A),p.selectAll(m).each(function(e,n){var r=d3.select(this),a=r.absolutePosition(),o=[[a.left-y[0]-x[0],a.top-y[1]-x[1]],[a.right-y[0]-x[0],a.top-y[1]-x[1]],[a.left-y[0]-x[0],a.bottom-y[1]-x[1]],[a.right-y[0]-x[0],a.bottom-y[1]-x[1]]];void 0!=S&&(o[0][0]=S.invert(o[0][0]),o[1][0]=S.invert(o[1][0]),o[2][0]=S.invert(o[2][0]),o[3][0]=S.invert(o[3][0])),void 0!=k&&(o[0][1]=k.invert(o[0][1]),o[1][1]=k.invert(o[1][1]),o[2][1]=k.invert(o[2][1]),o[3][1]=k.invert(o[3][1]));for(var i=!1,n=0;nd3.polygonContains(l,t))&&(i=!0)}r.classed("in-lasso",i),r.classed("in-lasso-"+E,i)}),f(),p.selectAll(".in-lasso-"+E)}function f(){p.selectAll(m).each(function(t,e){var n=d3.select(this);d(n,n.classed("in-lasso"))})}function d(t,e){var n=t.attr("_pre_lasso_fill"),r=t.attr("_pre_lasso_stroke"),a=t.attr("_pre_lasso_stroke-width");e?(t.classed("in-lasso",!0),t.classed("in-lasso-"+E,!0),void 0==n&&t.attr("_pre_lasso_fill",t.attr("fill")),void 0==r&&t.attr("_pre_lasso_stroke",t.attr("stroke")),void 0==a&&t.attr("_pre_lasso_stroke-width",t.attr("stroke-width")),t.attr("fill",V).attr("stroke",D).attr("stoke-width",K)):(t.classed("in-lasso",!1),t.classed("in-lasso-"+E,!1),void 0!=n&&t.attr("fill",n),void 0!=r&&t.attr("stroke",r),void 0!=a&&t.attr("stroke-width",a))}function h(){d3.select("html").select("style."+xt.str.hypenate(j,"lasso-dash")).empty()&&d3.select("html").append("style").classed(xt.str.hypenate(j,"lasso-dash"),!0).html("@keyframes lassoDash {to { stroke-dashoffset: 1000;}}")}var g,p,m,v,y,x,b,S,k,w,j="d3sm-lasso",z=!1,O=[],A=[],C=d3.line().x(function(t,e){return void 0!=S?S(t[0]):t[0]}).y(function(t,e){return void 0!=k?k(t[1]):t[1]}).curve(d3.curveLinearClosed),E=0,L=10,F="#17a2b8",M="10s",Q=.3,q="5, 10",B="black",W=2,V="white",D="black",K=3,_=1e3,P=d3.easeExp;return e.svg=function(t){return arguments.length?(g=t,e):g},e.chartContainer=function(t){return arguments.length?(v=t,e):v},e.objectContainer=function(t){return arguments.length?(p=t,e):p},e.objectClass=function(t){return arguments.length?(m=t,e):m},e.namespace=function(t){return arguments.length?(j=t,e):j},e.xScale=function(t){return arguments.length?(S=t,e):S},e.yScale=function(t){return arguments.length?(k=t,e):k},e.activeQ=function(t){return arguments.length?(z=t,e):z},e.currentPoints=function(t){return arguments.length?(O=t,e):O},e.allPoints=function(t){return arguments.length?(A=t,e):A},e.instance=function(t){return arguments.length?(E=t,e):E},e.tickDistance=function(t){return arguments.length?(L=t,e):L},e.color=function(t){return arguments.length?(F=t,e):F},e.animationRate=function(t){return arguments.length?(M=t,e):M},e.opacity=function(t){return arguments.length?(Q=t,e):Q},e.dashArray=function(t){return arguments.length?(q=t,e):q},e.stroke=function(t){return arguments.length?(B=t,e):B},e.lassoedFill=function(t){return arguments.length?(V=t,e):V},e.lassoedStroke=function(t){return arguments.length?(D=t,e):D},e.lassoedStrokeWidth=function(t){return arguments.length?(K=t,e):K},e.eventCatcher=function(t){return arguments.length?(b=t,e):b},e.drag=c,e.draw=r,e.tick=s,e.detect=u,e.toggle=n,e.remove=a,e.render=o,e.keyFrames=h,e.updateObjects=f,e.applyPathAttributes=l,e.applyObjectAttributes=d,h(),e}function st(t){function e(t,e){console.log(t,e)}function n(){d3.select("html").select("style."+V+"lasso-widget").empty()&&d3.select("html").append("style").classed(V+"lasso-widget",!0).html("."+xt.str.hypenate(V,"data-table")+"{ height:100px; overflow:auto; }")}function r(){n()}function a(t){console.log(t)}function o(t){var e=xt.sel.safeSelect(t,"li",xt.str.hypenate(V,"plus-tab")).classed("ml-auto",!0).classed("nav-item",!0).on("click",j),n=xt.sel.safeSelect(e,"a","nav-link");xt.sel.safeSelect(n,"i","fa").classed("fa-plus fa-2x text-success",!0)}function i(t){var e=xt.sel.safeSelect(t,"li",xt.str.hypenate(V,"send-tab")).classed("ml-auto",!0).classed("nav-item",!0).on("click",l),n=xt.sel.safeSelect(e,"a","nav-link");xt.sel.safeSelect(n,"i","fa").classed("fa-paper-plane-o fa-2x text-primary",!0)}function l(){a(A())}function c(e){var n=xt.sel.safeSelect(e,"li",xt.str.hypenate(V,"close-tab")).classed("ml-auto",!0).classed("nav-item",!0).on("click",function(){d3.mouse(d3.select("html").node());t.select("div.card").style("position","relative").transition().duration(1e3).ease(d3.easeBack).style("left",window.outerWidth+"px").remove()}),r=xt.sel.safeSelect(n,"a","nav-link");xt.sel.safeSelect(r,"i","fa").classed("fa-window-close-o fa-2x text-danger",!0)}function s(t){var e=xt.sel.safeSelect(t,"button","remove-btn").classed("btn btn-danger",!0).on("click",u);xt.sel.safeSelect(e,"i","fa").classed("fa-trash-o",!0),xt.sel.safeSelect(e,"span").text("Remove group");return e}function u(e,n){var r=t.select("#"+xt.str.hypenate(V,"tab",e)),a=t.select("#"+xt.str.hypenate(V,"tab","pane",e));r.remove(),a.remove(),C(),O()}function f(e){t.select("."+xt.str.hypenate(V,"tab-content")).selectAll("div.tab-pane").each(function(t,n){d3.select(this).classed("active show",d3.select(this).attr("id")==e)})}function d(t,e){var n=t.insert("li",".right-aligned-tabs").classed("nav-item",!0).classed("alert",!0).classed("alert-secondary",!0).classed("alert-dismissable",!0).classed("fade",!0).classed("show",!0).attr("role","alert").attr("id",xt.str.hypenate(V,"tab",e)).classed(xt.str.hypenate(V,"tab"),!0);xt.sel.safeSelect(n,"a").attr("data-toggle","tab").text("Group "+e).attr("href","#"+xt.str.hypenate(V,"tab","pane",e)).on("dblclick",function(t,e){var n=d3.select(this);n.attr("contenteditable",!0),d3.select(n.node().parentNode).classed("alert-secondary",!1).classed("alert-warning",!0)}).on("blur",function(t,e){var n=d3.select(this);n.attr("contenteditable",!1),d3.select(n.node().parentNode).classed("alert-secondary",!0).classed("alert-warning",!1)}).on("input",function(t,e){var n=d3.select(this),r=n.text();d3.select(n.attr("href")).select("p.lead").text(r)});return n}function h(t,e){return t.append("div").datum(e).attr("role","tabpanel").classed("tab-pane",!0).classed("text-left",!0).classed(xt.str.hypenate(V,"tab","pane"),!0).attr("id",xt.str.hypenate(V,"tab","pane",e))}function g(t,e){var n=xt.sel.safeSelect(t,"p","lead").text("Group "+e),r=xt.sel.safeSelect(t,"div","table-responsive").attr("class",xt.str.hypenate(V,"data-table")),a=xt.sel.safeSelect(r,"table","table").classed("table-sm",!0).classed("table-hover",!0),o=(xt.sel.safeSelect(a,"caption","caption").html("List of selected"),xt.sel.safeSelect(t,"div","text-right")),i=e%D.colors().length;e%2!=0&&(i=D.colors().length-1-i);var l=p(o),c=v(e,D.colors()[i]);y(l,m(o),c,a);s(o);n.on("mouseover",function(){c.draw()}),n.on("mouseout",function(){c.remove()})}function p(t){var e=xt.sel.safeSelect(t,"button","lasso-btn").classed("btn btn-info",!0).classed(V,!0);xt.sel.safeSelect(e,"i","fa").classed("fa-hand-pointer-o",!0),xt.sel.safeSelect(e,"span").text("Lasso select");return e}function m(t){var e=xt.sel.safeSelect(t,"button","clear-btn").classed("btn btn-light",!0).classed(V,!0);xt.sel.safeSelect(e,"i","fa").classed("fa-eraser",!0),xt.sel.safeSelect(e,"span").text("Clear selection");return e}function v(t,e){return ct().namespace(V).svg(E).objectClass(M).chartContainer(L).objectContainer(F).instance(t).color(e).yScale(W).xScale(B)}function y(t,e,n,r){n.eventCatcher(t),t.node().addEventListener("click",x),t.datum([n]).on(xt.str.hypenate(V,"render"),function(){d3.select(this).datum()[0].draw()}).on(xt.str.hypenate(V,"drag"),function(t){var e=L.selectAll(".in-lasso-"+t[0].instance()).nodes().map(function(t,e){var n=q(d3.select(t).datum());return n.__node=t,n});w(r,e,n)}),e.on("click",function(){n.allPoints([]),n.currentPoints([]),t.dispatch(xt.str.hypenate(V,"drag"))})}function x(){function e(t){t.target!=n.node()&&t.target!=n.select("span").node()&&t.target!=n.select("i").node()&&(r.toggle(!1),n.classed("btn-info",!0),n.classed("btn-warning",!1),n.select("span").text("Lasso select"))}var n=d3.select(this),r=n.datum()[0];r.toggle();var a=r.activeQ();r.activeQ()?(n.classed("btn-info",!a),n.classed("btn-warning",a),n.select("span").text("Lasso select (active)"),t.selectAll("."+V+".lasso-btn").dispatch(xt.str.hypenate(V,"render")),d3.select("html").node().addEventListener("mousedown",e)):(n.classed("btn-info",!a),n.classed("btn-warning",a),n.select("span").text("Lasso select"),d3.select("html").node().removeEventListener("mousedown",e))}function b(t,e){var n=d3.keys(e[0]).filter(t=>"__node"!=t);n.length>0&&n.push("remove");var r=t.selectAll("th");return(r=r.data(n)).exit().remove(),r=r.merge(r.enter().append("th").attr("scope","col")).text(function(t,e){return t}),n}function S(t,e){var n=t.selectAll("tr");return(n=n.data(e)).exit().remove(),n=n.merge(n.enter().append("tr"))}function k(t,e,n,r,a,o){(t=t.data(e)).exit().remove(),(t=t.merge(t.enter().append("td"))).html(function(t,e){return"remove"!=t?n[t]:""}).classed("text-left",function(t,e){return"remove"==t}),t.select("i.fa-close").on("click",function(t,e){var i=n.__node,l=d3.select(i);l.classed("in-lasso",!1),l.classed("in-lasso-"+a.instance(),!1),r.map(function(t,e){t.__node==i&&(r.splice(e,1),w(o,r,a))})})}function w(t,e,n){var r=xt.sel.safeSelect(t,"thead"),a=xt.sel.safeSelect(r,"tr"),o=xt.sel.safeSelect(t,"tbody"),i=b(a,e);S(o,e).each(function(r,a){var o=d3.select(this);k(o.selectAll("td"),i,r,e,n,t),o.on("mouseover",function(t,e){n.applyObjectAttributes(d3.select(t.__node),!0)}).on("mouseout",function(t,e){n.applyObjectAttributes(d3.select(t.__node),!1)})})}function j(n,r){var a=t.select("."+xt.str.hypenate(V,"tab-list")),o=t.select("."+xt.str.hypenate(V,"tab-content")),i=z();if(a.selectAll("."+xt.str.hypenate(V,"tab")).size()==Q)return void e("only "+Q+" allowed.","warning");d(a,i);var l=h(o,i);g(l,i),f(l.attr("id"))}function z(){var e=t.select("."+xt.str.hypenate(V,"tab-list")),n=e.selectAll("li").size()-3,r="Group "+n,a=[];for(e.each(function(t,e){return a.push(d3.select(this).select("a").text())});a.includes(r);)r="Group "+(n+=1);return n}function O(){var e=t.select("."+xt.str.hypenate(V,"tab-list")),n=t.select("."+xt.str.hypenate(V,"tab-content"));e=e.selectAll("."+xt.str.hypenate(V,"tab")),n=n.selectAll("."+xt.str.hypenate(V,"tab","pane")),e.each(function(t,e){d3.select(this).datum(e).attr("id",xt.str.hypenate(V,"tab",e)).select("a").attr("href","#"+xt.str.hypenate(V,"tab","pane",e)).text(function(t,n){var r=d3.select(this).text();return"Group"==r.split(" ")[0]?"Group "+e:r})}),n.each(function(t,e){d3.select(this).datum(e).attr("id",xt.str.hypenate(V,"tab","pane",e)),xt.sel.safeSelect(d3.select(this),"p","lead").text(function(t,n){var r=d3.select(this).text();return"Group"==r.split(" ")[0]?"Group "+e:r}),xt.sel.safeSelect(d3.select(this),"button","remove-btn").on("click",u)})}function A(){var e=t.select("."+xt.str.hypenate(V,"tab-list")),n=t.select("."+xt.str.hypenate(V,"tab-content")).selectAll("."+xt.str.hypenate(V,"data-table")),r={},a=e.selectAll("li."+xt.str.hypenate(V,"tab")+" > a").nodes().map(function(t,e){return d3.select(t).text()});return a.map(function(t,e){r[t]=[]}),n.each(function(t,e){var n=d3.select(this).select("tbody");r[a[e]]=n.selectAll("tr").data()}),r}function C(){var e=t.select("."+xt.str.hypenate(V,"tab-list")),n=t.select("."+xt.str.hypenate(V,"tab-content")),r=e.selectAll("."+xt.str.hypenate(V,"tab"));if(0==r.size())n.select("."+xt.str.hypenate(V,"default-tab")).classed("active",!0).classed("text-left",!0);else{var a=r.nodes()[r.size()-1],o=d3.select(a).select("a").attr("href");n.select(o).classed("active",!0).classed("text-left",!0)}}var E,L,F,M,Q,q,B,W,V="d3sm-lasso",t=t,D=(d3.line().x(function(t,e){return t[0]}).y(function(t,e){return t[1]}).curve(d3.curveLinearClosed),I());return function(){var e=xt.sel.safeSelect(t,"div","card"),n=xt.sel.safeSelect(e,"div","card-header"),r=xt.sel.safeSelect(n,"ul","nav").classed("nav-tabs card-header-tabs",!0).classed(xt.str.hypenate(V,"tab-list"),!0).attr("role","tablist"),a=xt.sel.safeSelect(e,"div","card-body"),l=xt.sel.safeSelect(a,"div","tab-content").classed(xt.str.hypenate(V,"tab-content"),!0),s=xt.sel.safeSelect(r,"div","right-aligned-tabs").classed("nav ml-auto ",!0);o(s),i(s),c(s);var u=xt.sel.safeSelect(l,"div","tab-pane").classed(xt.str.hypenate(V,"default-tab"),!0).classed("active",!0).classed("text-left",!0);xt.sel.safeSelect(u,"div").html("Click to add a new group.
Click to submit for re-analysis.
Click to close the dataselect widget.")}(),r.objectClass=function(t){return arguments.length?(M="."+t.replace(".",""),r):M},r.svg=function(t){return arguments.length?(E=t,r):E},r.submit=function(t){return arguments.length?(a=t,r):a},r.maxNumberOfGroups=function(t){return arguments.length?(Q=t,r):Q},r.onError=function(t){return arguments.length?(e=t,r):e},r.objectContainer=function(t){return arguments.length?(F=t,r):F},r.chartContainer=function(t){return arguments.length?(L=t,r):L},r.dataExtractor=function(t){return arguments.length?(q=t,r):q},r.xScale=function(t){return arguments.length?(B=t,r):B},r.yScale=function(t){return arguments.length?(W=t,r):W},r}function ut(t){function e(){var e=c.select("."+xt.str.hypenate(t.namespace(),"object-container")),n=xt.sel.getTranslation(e.attr("transform")),r=e.attr("transform","translate(0,0)");i=c.node().getBBox().width-t.spaceX(),l=c.node().getBBox().height-t.spaceY(),r.attr("transform","translate("+n[0]+","+n[1]+")"),xt.con.log("plotZoom","setLocks",{xLock:i,yLock:l})}function n(){e();var n,f,d=s.map(function(t,e){return t.selection()}),h=u.map(function(t,e){return t.selection()});"2D"==o&&(n=!0,f=!0),"horizontal"==o&&(n=!0,f=!1),"vertical"==o&&(f=!0,n=!1);var g=d3.event.transform,p=c.node().getBBox();d.map(function(t,e){return t.node().getBBox()}),d.map(function(t,e){return t.node().getBBox()}),p.width,p.x,p.height,p.y;if("wheel"==r){d3.event.preventDefault();var m=d3.event.deltaY*a,v=d3.event.shiftKey;(g="2D"==o?v?{k:1,x:m,y:0}:{k:1,x:0,y:m}:n?{k:1,x:m,y:0}:{k:1,x:0,y:m}).applyX=function(t){return t*this.k+-1*this.x},g.applyY=function(t){return t*this.k+-1*this.y}}var y=c.select("."+xt.str.hypenate(t.namespace(),"object-container")),x=d.map(function(t,e){return t.select("."+xt.str.hypenate(s[e].namespace(),"object-container"))}),b=h.map(function(t,e){return t.select("."+xt.str.hypenate(u[e].namespace(),"object-container"))}),S=xt.sel.getTranslation(y.attr("transform")),k=(x.map(function(t,e){return xt.sel.getTranslation(t.attr("transform"))}),b.map(function(t,e){return xt.sel.getTranslation(t.attr("transform"))}),n?g.applyX(S[0]):0);n&&(k=k<-i?(g.x=0,-i):(g.x=0,Math.min(k,0)));var w=f?g.applyY(S[1]):0;f&&(w=w<-l?(g.y=0,-l):(g.y=0,Math.min(w,0))),y.attr("transform","translate("+k+","+w+")"),n&&x.map(function(t,e){t.attr("transform","translate("+k+",0)")}),f&&b.map(function(t,e){t.attr("transform","translate(0,"+w+")")})}var r,a=20,o=void 0==t.orient?"horizontal":t.orient(),i=t.spaceX(),l=t.spaceY(),c=t.selection(),s=(d3.select(c.thisSVG()),[]),u=[];return n.eventType=function(t){return arguments.length?(r=t,n):r},n.wheelSpeed=function(t){return arguments.length?(a=t,n):a},n.orient=function(t){return arguments.length?(o=t,n):o},n.xLock=function(t){return arguments.length?(i=t,n):i},n.yLock=function(t){return arguments.length?(l=t,n):l},n.xComponents=function(t){return arguments.length?(s=t,n):s},n.yComponents=function(t){return arguments.length?(u=t,n):u},n.setLocks=e,n.reset=function(){var e=c.select("."+xt.str.hypenate(t.namespace(),"object-container")),n=xAxisSel.select("."+xt.str.hypenate(xAxis.namespace(),"object-container")),r=yAxisSel.select("."+xt.str.hypenate(yAxis.namespace(),"object-container"));e.attr("transform","translate(0,0)"),n.attr("transform","translate(0,0)"),r.attr("transform","translate(0,0)")},n}function ft(t,e,n){function r(){var e=u.select("."+xt.str.hypenate(t.namespace(),"object-container")),n=xt.sel.getTranslation(e.attr("transform")),r=e.attr("transform","translate(0,0)");c=u.node().getBBox().width-.9*t.spaceX(),s=u.node().getBBox().height-.9*t.spaceY(),r.attr("transform","translate("+n[0]+","+n[1]+")"),xt.con.log("plotZoom","setLocks",{xLock:c,yLock:s})}function a(){r();var a,h;"2D"==l&&(a=!0,h=!0),"horizontal"==l&&(a=!0,h=!1),"vertical"==l&&(h=!0,a=!1);var g=d3.event.transform,p=u.node().getBBox(),m=f.node().getBBox(),v=f.node().getBBox();p.width,p.x,p.height,p.y,m.width,m.height,v.width,v.height;if("wheel"==o){d3.event.preventDefault();var y=d3.event.deltaY*i,x=d3.event.shiftKey;(g="2D"==l?x?{k:1,x:y,y:0}:{k:1,x:0,y:y}:a?{k:1,x:y,y:0}:{k:1,x:0,y:y}).applyX=function(t){return t*this.k+-1*this.x},g.applyY=function(t){return t*this.k+-1*this.y}}var b=u.select("."+xt.str.hypenate(t.namespace(),"object-container")),S=f.select("."+xt.str.hypenate(e.namespace(),"object-container")),k=d.select("."+xt.str.hypenate(n.namespace(),"object-container")),w=xt.sel.getTranslation(b.attr("transform")),j=(xt.sel.getTranslation(S.attr("transform")),xt.sel.getTranslation(k.attr("transform")),a?g.applyX(w[0]):0);a&&(j=j<-c?(g.x=0,-c):(g.x=0,Math.min(j,0)));var z=h?g.applyY(w[1]):0;h&&(z=z<-s?(g.y=0,-s):(g.y=0,Math.min(z,0))),b.attr("transform","translate("+j+","+z+")"),a&&S.attr("transform","translate("+j+",0)"),h&&k.attr("transform","translate(0,"+z+")")}var o,i=20,l=void 0==t.orient?"horizontal":t.orient(),c=t.spaceX(),s=t.spaceY(),u=t.selection(),f=e.selection(),d=n.selection();d3.select(u.thisSVG());return a.eventType=function(t){return arguments.length?(o=t,a):o},a.wheelSpeed=function(t){return arguments.length?(i=t,a):i},a.orient=function(t){return arguments.length?(l=t,a):l},a.xLock=function(t){return arguments.length?(c=t,a):c},a.yLock=function(t){return arguments.length?(s=t,a):s},a.setLocks=r,a.reset=function(){var r=u.select("."+xt.str.hypenate(t.namespace(),"object-container")),a=f.select("."+xt.str.hypenate(e.namespace(),"object-container")),o=d.select("."+xt.str.hypenate(n.namespace(),"object-container"));r.attr("transform","translate(0,0)"),a.attr("transform","translate(0,0)"),o.attr("transform","translate(0,0)")},a}function dt(t){function e(){var e=xt.sel.safeSelect(t,"div","input-group").classed(xt.str.hypenate(a,"container"),!0),i=xt.sel.safeSelect(e,"div","select-prepend").classed("input-group-prepend",!0),c=(xt.sel.safeSelect(i,"span","input-group-text").text(o),xt.sel.safeSelect(e,"select","custom-select").classed(xt.str.hypenate(a,"select"),!0)),s=xt.sel.safeSelect(e,"div","select-append").classed("input-group-prepend",!0),u=xt.sel.safeSelect(s,"a","filter-button").classed("btn btn-outline-secondary",!0),f=(xt.sel.safeSelect(u,"i","fa fa-filter"),xt.sel.safeSelect(e,"div","filter-input-group").classed("input-group",!0).classed("d-none",!0)),d=xt.sel.safeSelect(f,"div","input-group-prepend"),h=xt.sel.safeSelect(d,"span","input-group-text").classed("search-button",!0),g=(xt.sel.safeSelect(h,"i","fa fa-search"),xt.sel.safeSelect(f,"input","form-control").attr("placeholder","all").attr("type","text")),p=xt.sel.safeSelect(f,"div","input-group-append"),m=xt.sel.safeSelect(p,"a","close-button").classed("btn btn-outline-secondary",!0),v=(xt.sel.safeSelect(m,"i","fa fa-close"),d3.keys(r)),y=c.selectAll("option");y=y.data(d3.keys(r)),y=y.merge(y.enter().append("option")).attr("value",function(t,e){return t}).text(function(t,e){return t});var x=m;u.on("click",function(t,e){var n=f.classed("d-none");f.classed("d-none",!n)}),x.on("click",function(t,e){g.property("value","").dispatch("input")}),g.on("input",function(t,e){var a,o=g.property("value"),i=new RegExp(o,"gi");""==o?a=v:(a=[],d3.keys(r).map(function(t,e){var n=t.match(i);null==n||""==n.join("")||a.push(t)})),y=c.selectAll("option"),(y=y.data(a)).exit().remove(),y=y.merge(y.enter().append("option")).attr("value",function(t,e){return t}).text(function(t,e){return t});var s=n();l!=s&&(l=s,c.dispatch("change"))})}function n(){var e=t.select("select").property("value");return void 0==e||""==e?void 0==i?d3.keys(r)[0]:i:e}var r,a="d3sm-select-filter",o="Select options:",i=void 0,l=void 0;return e.data=function(t){return arguments.length?(r=t,e):r},e.namespace=function(t){return arguments.length?(a=t,e):a},e.selectionName=function(t){return arguments.length?(o=t,e):o},e.defaultValue=function(t){return arguments.length?(i=t,e):i},e.currentOption=n,e}function ht(t){function e(){var n=xt.sel.safeSelect(t,"div","d-inline-flex flex-row flex-wrap").selectAll("div."+xt.str.hypenate(c,"select-filter"));n.exit().remove();var r=(n=n.data(d3.keys(i))).enter().append("div").attr("class","select-filter");return(n=n.merge(r).style("margin-right","10px")).each(function(t,e){var n=dt(d3.select(this)).data(i[t]).namespace(xt.str.hypenate(c,t)).selectionName(t);n(),f[t]=n}),t.selectAll("select").on("change",function(){l()}),e}var n,r,a,o,i,l=function(){},c="d3sm-databar",s=!1,u=!1;e.xAxisSelectQ=function(t){return arguments.length?(s=t,e):s},e.yAxisSelectQ=function(t){return arguments.length?(u=t,e):u},e.xAxisOptions=function(t){return arguments.length?(a=t,e):a},e.yAxisOptions=function(t){return arguments.length?(o=t,e):o},e.data=function(t){return arguments.length?(i=t,e):i},e.keys=function(t){return arguments.length?(n=t,e):n},e.currentKey=function(t){return arguments.length?(r=t,e):r},e.updateFunction=function(t){return arguments.length?(l=t,e):l},e.namespace=function(t){return arguments.length?(c=t,e):c},e.currentKeys=function(){var t={};return d3.keys(f).map(function(e,n){t[e]=f[e].currentOption()}),t};var f={};return e}var gt=Object.freeze({uniqueElements:e,all:n,tally:r,hasQ:a,first:o,last:i,total:l,unique:c,get:s,listOfListsQ:u,cut:f,groupBy:d,arrayEquals:h,elementsAtLevels:g,numberOfElements:p,flatten:m,whichBin:v}),pt=Object.freeze({modifyHexidecimalColorLuminance:y,interpolateColors:x}),mt=Object.freeze({round:b,tickRange:S,partitionRangeInto:k,euclideanDistance:w,quartiles:j,calculateWidthOfObject:z,calculateWidthOfSpacer:O,spacersNeededAtEachLevel:A}),vt=Object.freeze({hypenate:C,truncateText:E,truncateString:L}),yt=Object.freeze({getTranslation:F,getContainingSVG:M,safeSelect:Q,cpRect:q,bgRect:B,setupContainer:W});let xt={arr:gt,color:pt,math:mt,misc:Object.freeze({resizeDebounce:V,setupStandardChartContainers:K}),sel:yt,str:vt,con:Object.freeze({consoleGroup:_,consoleGroupEnd:P,log:R,warn:Y,info:X,error:T}),paths:Object.freeze({whiskerPath:N})},bt={scatter:J,bar:U,bubble:$,boxwhisker:tt,heatmap:et,violin:nt,neededViolinValues:rt,upset:at},St={categorical:ot,numeric:it};d3.selection.prototype.thisSVG=function(){return xt.sel.getContainingSVG(this.node())},d3.mouse.absolute=function(){var t=d3.select("html").node(),[e,n]=this(t);return[e,n]},d3.selection.prototype.absolutePosition=function(){var t=this.node(),e=t.getBoundingClientRect(),n=xt.sel.getContainingSVG(t).getBoundingClientRect();return{top:e.top-n.top,left:e.left-n.left,bottom:e.bottom-n.top,right:e.right-n.left,height:e.height,width:e.width}},d3.selection.prototype.relativePositionTo=function(t){var e=this.node().getBoundingClientRect(),n=t.getBoundingClientRect();return{top:e.top-n.top,left:e.left-n.left,bottom:e.bottom-n.top,right:e.right-n.left,height:e.height,width:e.width}};let kt={aux:{lasso:ct,plotZoom:ft,multiPlotZoom:ut,datatoggle:ht,selectFilter:dt,lassoWidget:st},axis:H,charts:bt,colorFunction:I,debugQ:!1,groupingSpacer:G,legends:St,tooltip:Z,utils:xt};"undefined"!=typeof window&&(window.d3sm=kt),t.default=kt}(this.d3sm=this.d3sm||{}); diff --git a/dist/d3sm.umd.js b/dist/d3sm.umd.js index 20f1408..2ea4d72 100644 --- a/dist/d3sm.umd.js +++ b/dist/d3sm.umd.js @@ -64,7 +64,7 @@ * @param {number[]} array of numerical values * @returns {number} sum over elements in array */ - function total$1( array ) { return array.reduce((a, b) => a + b, 0) } + function total( array ) { return array.reduce((a, b) => a + b, 0) } /** * Removes duplicates in array * @param {Array} array of items @@ -218,7 +218,7 @@ hasQ: hasQ, first: first, last: last, - total: total$1, + total: total, unique: unique, get: get, listOfListsQ: listOfListsQ, @@ -441,150 +441,6 @@ spacersNeededAtEachLevel: spacersNeededAtEachLevel }); - function resizeDebounce(f, wait) { - var resize = debounce(function(){f();},wait); - window.addEventListener('resize', resize); - } - - function debounce(func, wait, immediate) { - var timeout; - return function() { - var context = this, args = arguments; - var later = function() { - timeout = null; - if (!immediate) func.apply(context, args); - }; - var callNow = immediate && !timeout; - clearTimeout(timeout); - timeout = setTimeout(later, wait); - if (callNow) func.apply(context, args); - }; - } - - function setupStandardChartContainers( - selection, - namespace, - container, - margins={top:0.01, bottom:0.01, left:0.01, right:0.01}, - svg={w:0.8, h:0.6}, // percent of container space for svg - axes={y:0.1, x:0.1}, // percent of container space for axes, - leg={x:0, margin:0, pos:'left'} // absolute width of legend and space on either size - ) - { - if (container == undefined) {container = {w:window.innerWidth, h:window.Height};} - // SVG width and height - - var svgSpace = { - w: container.w * svg.w, - h: container.h * svg.h - }; - - var margPx = { - top: margins.top * svgSpace.h, - bottom: margins.bottom * svgSpace.h, - left: margins.left * svgSpace.w, - right: margins.right * svgSpace.w - }, - - - - // Space after removing margins - chartSpace = { - w: svgSpace.w - margPx.left - margPx.right, - h: svgSpace.h - margPx.top - margPx.bottom - }, - - // main dimension of x and y axies - // e.g. defines how tall x axis is as length is determined by plotRect.w - axesSpace = { - x: chartSpace.h * axes.x, - y: chartSpace.w * axes.y - }, - - // space left for drawing the chart properly (e.g. bars, violins, etc) - drawingSpace = { - x: chartSpace.w - axesSpace.y - leg.x - 2*leg.margin, - y: chartSpace.h - axesSpace.x - }, - - - legRect = { - x: leg.margin + margPx.left + (leg.pos == 'left' ? 0 : drawingSpace.x + axesSpace.y), - y: margPx.top, // this is soomehow getting calculated incorectly - w: leg.x, - h: drawingSpace.y - }, - - yAxisRect = { - x: axesSpace.y + margPx.left + (leg.pos == 'left' ? leg.x + 2*leg.margin : 0), - y: margPx.top, - w: axesSpace.y, - h: drawingSpace.y - }, - - plotRect = { - x: axesSpace.y + margPx.left + (leg.pos == 'left' ? leg.x + 2*leg.margin : 0), - y: margPx.top, - w: drawingSpace.x, - h: drawingSpace.y - }, - - xAxisRect = { - x: axesSpace.y + margPx.left + (leg.pos == 'left' ? leg.x + 2*leg.margin : 0), - y: margPx.top + drawingSpace.y, - w: drawingSpace.x, - h: axesSpace.x - }; - - - - container = d3sm.safeSelect(selection, 'svg', namespace) - .style('width', svgSpace.w+'px') - .style('height', svgSpace.h+'px'); - - var axes = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'axes')); - - var leg = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'legend')) - .attr('transform', "translate("+legRect.x+","+legRect.y+")"); - - var plot = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'plot')) - .attr('transform', "translate("+plotRect.x+","+plotRect.y+")"); - - var xAxis = d3sm.safeSelect(axes, 'g', d3sm.hypenate(namespace, 'x-axis')) - .attr('transform', "translate("+xAxisRect.x+","+xAxisRect.y+")"); - - var yAxis = d3sm.safeSelect(axes, 'g', d3sm.hypenate(namespace, 'y-axis')) - .attr('transform', "translate("+yAxisRect.x+","+yAxisRect.y+")"); - - return { - svg: { - selection: container, - rect: svgSpace - }, - plot: { - selection: plot, - rect: plotRect - }, - xAxis: { - selection: xAxis, - rect: xAxisRect - }, - yAxis: { - selection: yAxis, - rect: yAxisRect - }, - legend: { - selection: leg, - rect: legRect - } - } - } - - var misc = /*#__PURE__*/Object.freeze({ - resizeDebounce: resizeDebounce, - setupStandardChartContainers: setupStandardChartContainers - }); - /** * Hypenates all strings together * @param {string[]} arguments @@ -658,12 +514,12 @@ * @param {Element} element * @returns {Element | undefined} */ - function getContainingSVG$1(element) { + function getContainingSVG(element) { var parent = element.parentElement; var tag = parent.tagName.toLowerCase(); if (tag === 'svg') { return parent; } if (tag === 'html') { return undefined; } - return getContainingSVG$1(parent); + return getContainingSVG(parent); } @@ -754,100 +610,244 @@ var sel = /*#__PURE__*/Object.freeze({ getTranslation: getTranslation, - getContainingSVG: getContainingSVG$1, + getContainingSVG: getContainingSVG, safeSelect: safeSelect, cpRect: cpRect, bgRect: bgRect, setupContainer: setupContainer }); - /** - * calls console.group if d3sm.debugQ == true - * @param {string} name of the group - * @returns {undefined} - */ - function consoleGroup(name) { - if (window.d3sm.debugQ === true){ - console.group(name); - } + function resizeDebounce(f, wait) { + var resize = debounce(function(){f();},wait); + window.addEventListener('resize', resize); } - /** - * calls console.groupEnd if d3sm.debugQ == true - * @returns {undefined} - */ - function consoleGroupEnd() { - if (window.d3sm.debugQ === true){ - console.groupEnd(); - } + function debounce(func, wait, immediate) { + var timeout; + return function() { + var context = this, args = arguments; + var later = function() { + timeout = null; + if (!immediate) func.apply(context, args); + }; + var callNow = immediate && !timeout; + clearTimeout(timeout); + timeout = setTimeout(later, wait); + if (callNow) func.apply(context, args); + }; } - /** - * Calls console.log if d3sm.debugQ == true - * @param {string} func name of the function logging - * @param {string} msg to log - * @param {Object} data to be logged along side the message - * @returns {undefined} - */ - function log(func, msg, data) { - if (window.d3sm.debugQ === true){ - console.log( - `%c[d3sm::${func}]:\t${msg}`, - [ - 'background: #6cd1ef', - 'border-radius: 5000px', - 'padding: 0px 2px', - 'font-size: 14px' - ].join(';') - ); - console.table(data); - // console.trace() - } - } + function setupStandardChartContainers( + selection, + namespace, + container, + margins={top:0.01, bottom:0.01, left:0.01, right:0.01}, + svg={w:0.8, h:0.6}, // percent of container space for svg + axes={y:0.1, x:0.1}, // percent of container space for axes, + leg={x:0, margin:0, pos:'left'} // absolute width of legend and space on either size + ) + { + if (container == undefined) {container = {w:window.innerWidth, h:window.Height};} + // SVG width and height - /** - * Calls console.warn if d3sm.debugQ == true - * @param {string} func name of the function warning - * @param {string} msg to display - * @param {Object} data to be displayed along side the message - * @returns {undefined} - */ - function warn(func, msg, data) { - if (window.d3sm.debugQ === true) - console.warn( - `%c[d3sm::${func}]:\t${msg}`, - [ - 'background: #ffd53e', - 'border-radius: 5000px', - 'padding: 0px 2px', - 'font-size: 14px' - ].join(';') - ); - console.table(data); - } - /** - * Calls the console.info if d3sm.debugQ == true - * @param {string} func name of the function providing info - * @param {string} msg to display - * @param {Object} data to be displayed along side the message - * @returns {undefined} - */ - function info(func, msg, data) { - if (window.d3sm.debugQ) - console.info( - `%c[d3sm::${func}]:\t${msg}`, - [ - 'background: #009ccd', - 'border-radius: 5000px', - 'padding: 0px 2px', - 'font-size: 14px' - ].join(';') - ); - console.table(data); - } + var svgSpace = { + w: container.w * svg.w, + h: container.h * svg.h + }; + var margPx = { + top: margins.top * svgSpace.h, + bottom: margins.bottom * svgSpace.h, + left: margins.left * svgSpace.w, + right: margins.right * svgSpace.w + }, - /** + + + // Space after removing margins + chartSpace = { + w: svgSpace.w - margPx.left - margPx.right, + h: svgSpace.h - margPx.top - margPx.bottom + }, + + // main dimension of x and y axies + // e.g. defines how tall x axis is as length is determined by plotRect.w + axesSpace = { + x: chartSpace.h * axes.x, + y: chartSpace.w * axes.y + }, + + // space left for drawing the chart properly (e.g. bars, violins, etc) + drawingSpace = { + x: chartSpace.w - axesSpace.y - leg.x - 2*leg.margin, + y: chartSpace.h - axesSpace.x + }, + + + legRect = { + x: leg.margin + margPx.left + (leg.pos == 'left' ? 0 : drawingSpace.x + axesSpace.y), + y: margPx.top, // this is soomehow getting calculated incorectly + w: leg.x, + h: drawingSpace.y + }, + + yAxisRect = { + x: axesSpace.y + margPx.left + (leg.pos == 'left' ? leg.x + 2*leg.margin : 0), + y: margPx.top, + w: axesSpace.y, + h: drawingSpace.y + }, + + plotRect = { + x: axesSpace.y + margPx.left + (leg.pos == 'left' ? leg.x + 2*leg.margin : 0), + y: margPx.top, + w: drawingSpace.x, + h: drawingSpace.y + }, + + xAxisRect = { + x: axesSpace.y + margPx.left + (leg.pos == 'left' ? leg.x + 2*leg.margin : 0), + y: margPx.top + drawingSpace.y, + w: drawingSpace.x, + h: axesSpace.x + }; + + + + container = safeSelect(selection, 'svg', namespace) + .style('width', svgSpace.w+'px') + .style('height', svgSpace.h+'px'); + + var axes = safeSelect(container, 'g', hypenate(namespace, 'axes')); + + var leg = safeSelect(container, 'g', hypenate(namespace, 'legend')) + .attr('transform', "translate("+legRect.x+","+legRect.y+")"); + + var plot = safeSelect(container, 'g', hypenate(namespace, 'plot')) + .attr('transform', "translate("+plotRect.x+","+plotRect.y+")"); + + var xAxis = safeSelect(axes, 'g', hypenate(namespace, 'x-axis')) + .attr('transform', "translate("+xAxisRect.x+","+xAxisRect.y+")"); + + var yAxis = safeSelect(axes, 'g', hypenate(namespace, 'y-axis')) + .attr('transform', "translate("+yAxisRect.x+","+yAxisRect.y+")"); + + return { + svg: { + selection: container, + rect: svgSpace + }, + plot: { + selection: plot, + rect: plotRect + }, + xAxis: { + selection: xAxis, + rect: xAxisRect + }, + yAxis: { + selection: yAxis, + rect: yAxisRect + }, + legend: { + selection: leg, + rect: legRect + } + } + } + + var misc = /*#__PURE__*/Object.freeze({ + resizeDebounce: resizeDebounce, + setupStandardChartContainers: setupStandardChartContainers + }); + + /** + * calls console.group if d3sm.debugQ == true + * @param {string} name of the group + * @returns {undefined} + */ + function consoleGroup(name) { + if (window.d3sm.debugQ === true){ + console.group(name); + } + } + + /** + * calls console.groupEnd if d3sm.debugQ == true + * @returns {undefined} + */ + function consoleGroupEnd() { + if (window.d3sm.debugQ === true){ + console.groupEnd(); + } + } + + /** + * Calls console.log if d3sm.debugQ == true + * @param {string} func name of the function logging + * @param {string} msg to log + * @param {Object} data to be logged along side the message + * @returns {undefined} + */ + function log(func, msg, data) { + if (window.d3sm.debugQ === true){ + console.log( + `%c[d3sm::${func}]:\t${msg}`, + [ + 'background: #6cd1ef', + 'border-radius: 5000px', + 'padding: 0px 2px', + 'font-size: 14px' + ].join(';') + ); + console.table(data); + // console.trace() + } + } + + /** + * Calls console.warn if d3sm.debugQ == true + * @param {string} func name of the function warning + * @param {string} msg to display + * @param {Object} data to be displayed along side the message + * @returns {undefined} + */ + function warn(func, msg, data) { + if (window.d3sm.debugQ === true) + console.warn( + `%c[d3sm::${func}]:\t${msg}`, + [ + 'background: #ffd53e', + 'border-radius: 5000px', + 'padding: 0px 2px', + 'font-size: 14px' + ].join(';') + ); + console.table(data); + } + /** + * Calls the console.info if d3sm.debugQ == true + * @param {string} func name of the function providing info + * @param {string} msg to display + * @param {Object} data to be displayed along side the message + * @returns {undefined} + */ + function info(func, msg, data) { + if (window.d3sm.debugQ) + console.info( + `%c[d3sm::${func}]:\t${msg}`, + [ + 'background: #009ccd', + 'border-radius: 5000px', + 'padding: 0px 2px', + 'font-size: 14px' + ].join(';') + ); + console.table(data); + } + + + /** * Calls console.error if d3sm.debugQ == true * @param {string} func name of the function which sends the error * @param {string} msg to display @@ -912,7 +912,7 @@ whiskerPath: whiskerPath }); - let utils$1 = { + let utils = { arr, color, math, misc, sel, str, con, paths }; @@ -2022,7 +2022,7 @@ function axis () { // for convenience in handling orientation specific values - var horizontalQ = utils$1.arr.hasQ(['top', 'bottom', 'horizontal'], orient) ? true : false; + var horizontalQ = utils.arr.hasQ(['top', 'bottom', 'horizontal'], orient) ? true : false; var verticalQ = !horizontalQ; // background cliping rectangle @@ -2053,7 +2053,7 @@ } - var container = utils$1.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); + var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); // defaults for text-anchor and text rotation if (orient == 'top') { @@ -2091,12 +2091,12 @@ : (grouping == undefined) ? (numberOfTicks != undefined) // ? (tickValues.length < numberOfTicks) - ? (utils$1.math.tickRange(...d3.extent(tickValues), numberOfTicks)) + ? (utils.math.tickRange(...d3.extent(tickValues), numberOfTicks)) : tickValues : grouping; - var flatTickData = utils$1.arr.flatten(tickData); + var flatTickData = utils.arr.flatten(tickData); var numberOfObjects = flatTickData.length; var space = horizontalQ ? spaceX : spaceY; var extent = d3.extent(flatTickData); @@ -2120,16 +2120,16 @@ // calculate object size if needed objectSize = (objectSize == undefined) - ? utils$1.math.calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ) + ? utils.math.calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ) : objectSize; // calculate spacer size if needed spacerSize = (spacerSize == undefined) - ? utils$1.math.calculateWidthOfSpacer(flatTickData, space, objectSize, numberOfObjects, objectSpacer, overflowQ) + ? utils.math.calculateWidthOfSpacer(flatTickData, space, objectSize, numberOfObjects, objectSpacer, overflowQ) : spacerSize; - var objClass = utils$1.str.hypenate(namespace, categoricalQ ? objectClass+'-categorical' : objectClass); + var objClass = utils.str.hypenate(namespace, categoricalQ ? objectClass+'-categorical' : objectClass); var spacerFunction = groupingSpacer() .horizontalQ(horizontalQ).scale(scale).moveby((categoricalQ?'category':'scale')).numberOfObjects(numberOfObjects) @@ -2194,11 +2194,11 @@ - var labelNameGroup = utils$1.sel.safeSelect(selection, 'g', utils$1.str.hypenate(namespace,'axis-name')); + var labelNameGroup = utils.sel.safeSelect(selection, 'g', utils.str.hypenate(namespace,'axis-name')); - var labelElement = utils$1.sel.safeSelect(labelNameGroup, 'text', utils$1.str.hypenate(namespace, 'name')); + var labelElement = utils.sel.safeSelect(labelNameGroup, 'text', utils.str.hypenate(namespace, 'name')); if (labelElement != undefined) { labelElement.text(label); @@ -2263,7 +2263,7 @@ var that = d3.select(this).style('opacity', 1); // make and move tick - var tick = utils$1.sel.safeSelect(that, 'line', utils$1.str.hypenate(namespace,'tick')) + var tick = utils.sel.safeSelect(that, 'line', utils.str.hypenate(namespace,'tick')) .attr("x1", 0) .attr("x2", horizontalQ ? 0 : orient == "left" ? -tickLength : tickLength) .attr("y1", 0) @@ -2279,10 +2279,10 @@ }); // make and move label - var label = utils$1.sel.safeSelect(that, 'text', utils$1.str.hypenate(namespace,'label')) + var label = utils.sel.safeSelect(that, 'text', utils.str.hypenate(namespace,'label')) .text(function(d, i){ - var s = typeof d == 'number' ? utils$1.math.round(d, roundTo) : d; - s = utils$1.str.truncateString(String(s), (horizontalQ ? spaceY : spaceX) - tickLength-tickLabelMargin-tickTickLabelSpacer, tickLabelFontSize * 0.45); + var s = typeof d == 'number' ? utils.math.round(d, roundTo) : d; + s = utils.str.truncateString(String(s), (horizontalQ ? spaceY : spaceX) - tickLength-tickLabelMargin-tickTickLabelSpacer, tickLabelFontSize * 0.45); return s }) .attr('font-size', tickLabelFontSize) @@ -2334,7 +2334,7 @@ // add guidlines as needed if (guideLinesQ) { - var gline = utils$1.sel.safeSelect(that, 'line', utils$1.str.hypenate(namespace, 'guideline')) + var gline = utils.sel.safeSelect(that, 'line', utils.str.hypenate(namespace, 'guideline')) .transition().duration(transitionDuration).ease(easeFunc) .attr("x1", 0) .attr("x2", horizontalQ ? 0 : orient == "left" ? guidelineSpace : -guidelineSpace) @@ -2347,15 +2347,15 @@ t = 'translate('+x+','+y+')'; return t }); - } else { that.select('line.'+utils$1.str.hypenate(namespace, 'guideline')).remove(); } + } else { that.select('line.'+utils.str.hypenate(namespace, 'guideline')).remove(); } }); // apply alternating guidline thickness if (guideLinesQ) { - container.selectAll('.'+utils$1.str.hypenate(namespace,'guideline')) + container.selectAll('.'+utils.str.hypenate(namespace,'guideline')) .attr('stroke', function(d, i){ - if (i % 2 == 0) { return utils$1.color.modifyHexidecimalColorLuminance(guideLineStroke, 0.8) } + if (i % 2 == 0) { return utils.color.modifyHexidecimalColorLuminance(guideLineStroke, 0.8) } return guideLineStroke }) .attr('stroke-width', function(d, i){ @@ -2369,7 +2369,7 @@ /*************************************************************************** ** Make the line of the axis ***************************************************************************/ - var line = utils$1.sel.safeSelect(selection, 'path', utils$1.str.hypenate(namespace,'line')) + var line = utils.sel.safeSelect(selection, 'path', utils.str.hypenate(namespace,'line')) // .attr('x1', 0) // .attr('x2', horizontalQ ? spaceX : 0) // .attr('y1', 0) @@ -2389,21 +2389,21 @@ // hover of label show full text label in case it is truncated function labelHover(d, i){ var t = d3.select(this).style('fill', 'red'); - d3.select(t.node().parentNode).select("line."+utils$1.str.hypenate(namespace,'tick')) + d3.select(t.node().parentNode).select("line."+utils.str.hypenate(namespace,'tick')) .attr("stroke", 'red') .attr("stroke-width", tickStrokeWidth*2); if (guideLinesQ) { - d3.select(t.node().parentNode).select('line.'+utils$1.str.hypenate(namespace, 'guideline')) + d3.select(t.node().parentNode).select('line.'+utils.str.hypenate(namespace, 'guideline')) .attr('stroke', 'red') .attr('stroke-width', guideLineStrokeWidth*2); } - var s = typeof d == 'number' ? utils$1.math.round(d, roundTo) : d; + var s = typeof d == 'number' ? utils.math.round(d, roundTo) : d; var m = d3.mouse(d3.select('html').node()); - var div = utils$1.sel.safeSelect(d3.select('body'), 'div', utils$1.str.hypenate(namespace,'guideline-tooltip')) - .attr('id', utils$1.str.hypenate(namespace,'guideline-tooltip')) + var div = utils.sel.safeSelect(d3.select('body'), 'div', utils.str.hypenate(namespace,'guideline-tooltip')) + .attr('id', utils.str.hypenate(namespace,'guideline-tooltip')) .style('position', 'absolute') .style('left', (d3.event.pageX+15)+'px') .style('top', (d3.event.pageY+15)+'px') @@ -2420,7 +2420,7 @@ .style('border-style', 'solid') .style('border-width', 2); - var text = utils$1.sel.safeSelect(div, 'div') + var text = utils.sel.safeSelect(div, 'div') .text(tickLabelOnHoverFunc(s, i)) .style('color', 'black') .style('align-self', 'center'); @@ -2433,15 +2433,15 @@ function labelHoverOff(d, i){ var t = d3.select(this).style('fill', 'black'); - d3.select(t.node().parentNode).select("line."+utils$1.str.hypenate(namespace,'tick')) + d3.select(t.node().parentNode).select("line."+utils.str.hypenate(namespace,'tick')) .attr("stroke", tickStroke) .attr("stroke-width", tickStrokeWidth); if (guideLinesQ) { - var gline = d3.select(t.node().parentNode).select('line.'+utils$1.str.hypenate(namespace, 'guideline')); + var gline = d3.select(t.node().parentNode).select('line.'+utils.str.hypenate(namespace, 'guideline')); var minorQ = gline.attr('minor'); gline.attr('stroke', function(d, ii){ - if (minorQ == 'true') { return utils$1.color.modifyHexidecimalColorLuminance(guideLineStroke, 0.8) } + if (minorQ == 'true') { return utils.color.modifyHexidecimalColorLuminance(guideLineStroke, 0.8) } return guideLineStroke }) .attr('stroke-width', function(d, ii){ @@ -2449,7 +2449,7 @@ return guideLineStrokeWidth }); } - d3.select("#"+utils$1.str.hypenate(namespace,'guideline-tooltip')).remove(); + d3.select("#"+utils.str.hypenate(namespace,'guideline-tooltip')).remove(); } @@ -3396,7 +3396,7 @@ function scatter() { // background cliping rectangle var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; - var container = utils$1.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); + var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); pointKeys = d3.keys(data); @@ -3975,7 +3975,7 @@ // background cliping rectangle var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; - var container = utils$1.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); + var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); // to prevent re-calculation and getters to be passed to axes barKeys = d3.keys(data); @@ -3984,7 +3984,7 @@ // if grouping is undefined sort barKeys by sortingFunction var ordered = (grouping == undefined) ? barKeys.sort(sortingFunction) : grouping; // ordered might be nested depending on grouping - barKeys = utils$1.arr.flatten(ordered); + barKeys = utils.arr.flatten(ordered); var numberOfObjects = barKeys.length; var extent = [Math.min(...barValues) - domainPadding,Math.max(...barValues) + domainPadding]; @@ -4002,12 +4002,12 @@ var space = horizontalQ ? spaceX : spaceY; // calculate object size objectSize = (objectSize == undefined) - ? utils$1.math.calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ) + ? utils.math.calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ) : objectSize; // calculate spacer size if needed spacerSize = (spacerSize == undefined) - ? utils$1.math.calculateWidthOfSpacer(barKeys, space, objectSize, numberOfObjects, objectSpacer, overflowQ) + ? utils.math.calculateWidthOfSpacer(barKeys, space, objectSize, numberOfObjects, objectSpacer, overflowQ) : spacerSize; // make the nested groups var spacerFunction = groupingSpacer() @@ -4074,7 +4074,7 @@ strokeColor = colorFunction$$1(key, value, i, 'stroke'); - var bar = utils$1.sel.safeSelect(t, 'rect', 'bar-rect'); + var bar = utils.sel.safeSelect(t, 'rect', 'bar-rect'); if (bar.attr('transform') == undefined) { bar.attr('transform', function(d, i) { @@ -4804,19 +4804,19 @@ function bhm() { var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; - var container = utils$1.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); + var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); cellKeys = d3.keys(data); cellKeys.sort(function(a, b){ return xKeySortingFunction(a, b) || yKeySortingFunction(a, b) }); - utils$1.con.log('bubbleHeatmap', 'cells are sorted by', cellKeys); + utils.con.log('bubbleHeatmap', 'cells are sorted by', cellKeys); - xValues = utils$1.arr.unique(cellKeys.map(xExtractor)); - yValues = utils$1.arr.unique(cellKeys.map(yExtractor)); - rValues = utils$1.arr.unique(cellKeys.map(rExtractor)); - vValues = utils$1.arr.unique(cellKeys.map(vExtractor)); - utils$1.con.log('bubbleHeatmap', 'x and y keys are', {x: xValues, y:yValues}); + xValues = utils.arr.unique(cellKeys.map(xExtractor)); + yValues = utils.arr.unique(cellKeys.map(yExtractor)); + rValues = utils.arr.unique(cellKeys.map(rExtractor)); + vValues = utils.arr.unique(cellKeys.map(vExtractor)); + utils.con.log('bubbleHeatmap', 'x and y keys are', {x: xValues, y:yValues}); var xDim = xValues.length, yDim = yValues.length; @@ -4825,11 +4825,11 @@ var extent = [Math.min(...rValues) - domainPadding,Math.max(...rValues) + domainPadding]; - ySize = utils$1.math.calculateWidthOfObject(spaceY, yDim, minObjectSize, maxObjectSize, objectSpacer, overflowQ); - xSize = utils$1.math.calculateWidthOfObject(spaceX, xDim, minObjectSize, maxObjectSize, objectSpacer, overflowQ); - ySpacerSize = utils$1.math.calculateWidthOfSpacer(yValues, spaceY, ySize, yDim, objectSpacer, overflowQ); - xSpacerSize = utils$1.math.calculateWidthOfSpacer(xValues, spaceX, xSize, xDim, objectSpacer, overflowQ); - utils$1.con.log('bubbleHeatmap', 'size of', {x: xSize, y: ySize}); + ySize = utils.math.calculateWidthOfObject(spaceY, yDim, minObjectSize, maxObjectSize, objectSpacer, overflowQ); + xSize = utils.math.calculateWidthOfObject(spaceX, xDim, minObjectSize, maxObjectSize, objectSpacer, overflowQ); + ySpacerSize = utils.math.calculateWidthOfSpacer(yValues, spaceY, ySize, yDim, objectSpacer, overflowQ); + xSpacerSize = utils.math.calculateWidthOfSpacer(xValues, spaceX, xSize, xDim, objectSpacer, overflowQ); + utils.con.log('bubbleHeatmap', 'size of', {x: xSize, y: ySize}); scale.domain(extent).range([Math.min(minObjectSize/2, Math.min(ySize, xSize)/2), Math.min(ySize, xSize)/2]); @@ -4837,7 +4837,7 @@ var ySpacer = groupingSpacer() .horizontalQ(false) .moveby('category').numberOfObjects(yDim) - .objectClass(utils$1.str.hypenate(objectClass, 'row')) + .objectClass(utils.str.hypenate(objectClass, 'row')) .objectSize(ySize).spacerSize(ySpacerSize) .transitionDuration(transitionDuration).easeFunc(easeFunc) .namespace('row'); @@ -4851,7 +4851,7 @@ ySpacer(container, yValues, 0); - container.selectAll('g.'+utils$1.str.hypenate(objectClass, 'row')) + container.selectAll('g.'+utils.str.hypenate(objectClass, 'row')) .each(function(d, i){ xSpacer(d3.select(this), xValues, 0); }); @@ -4866,7 +4866,7 @@ : colorFunction$$1.dataExtent([0, Math.max(...vValues)]); cells.each(function(key, i) { - utils$1.con.log('bubbleHeatmap', 'each cell', {key: key, index: i, node: d3.select(this).node()}); + utils.con.log('bubbleHeatmap', 'each cell', {key: key, index: i, node: d3.select(this).node()}); var t = d3.select(this), currentData = data[key], @@ -4876,9 +4876,9 @@ fillColor = colorFunction$$1(key, value, i, 'fill'), // prevent duplicate computation strokeColor = colorFunction$$1(key, value, i, 'stroke'); - utils$1.con.log('bubbleHeatmap', 'radius',{radius: radius, scaled: scale(radius), extent: extent, range:scale.range()}); + utils.con.log('bubbleHeatmap', 'radius',{radius: radius, scaled: scale(radius), extent: extent, range:scale.range()}); - var c = utils$1.sel.safeSelect(t, 'circle', utils$1.str.hypenate(objectClass,'circle')); + var c = utils.sel.safeSelect(t, 'circle', utils.str.hypenate(objectClass,'circle')); c.attr('cx', xSize / 2) .attr('cy', ySize / 2 ) .attr('r', scale(radius)) @@ -4888,7 +4888,7 @@ }); - tooltip$$1.selection(cells.selectAll('circle.'+utils$1.str.hypenate(objectClass, 'circle'))) + tooltip$$1.selection(cells.selectAll('circle.'+utils.str.hypenate(objectClass, 'circle'))) .data(data); // .keys(['r', 'v']) // .header(function(d, i){return utils.str.hypenate(data[d][xKey], data[d][yKey]) }) @@ -4901,3046 +4901,4368 @@ return bhm; } + /******************************************************************************* + ** ** + ** ** + ** BOX AND WHISKER ** + ** ** + ** ** + *******************************************************************************/ /** - * Creates a heatmap + * Creates a boxwhisker * - * {@link https://sumneuron.gitlab.io/d3sm/demos/heatmap-heatmap/index.html Demo} - * @constructor heatmap + * {@link https://sumneuron.gitlab.io/d3sm/demos/box-whiskers/index.html Demo} + * @constructor boxwhisker * @param {d3.selection} selection - * @namespace heatmap - * @returns {function} heatmap + * @namespace boxwhisker + * @returns {function} boxwhisker */ - function heatmap( selection ) { - var + function boxwhisker( selection ) { + var /** - * Data to plot. Assumed to be a object, where each key corresponds to a cell - * (see {@link heatmap#data}) + * Data to plot. Assumed to be a object, where each key corresponds to a box + * (see {@link boxwhisker#data}) * @param {Object} [data=undefined] - * @memberof heatmap# + * @memberof boxwhisker# * @property */ data, - /** - * Amount of horizontal space (in pixels) avaible to render the heatmap in - * (see {@link heatmap#spaceX}) + * Which direction to render the boxes in + * (see {@link boxwhisker#orient}) + * @param {number} [orient='horizontal'] + * @memberof boxwhisker# + * @property + */ + orient = 'horizontal', + /** + * Amount of horizontal space (in pixels) avaible to render the boxwhisker in + * (see {@link boxwhisker#spaceX}) * @param {number} [spaceX=undefined] - * @memberof heatmap# + * @memberof boxwhisker# * @property */ spaceX, /** - * Amount of vertical space (in pixels) avaible to render the heatmap in - * (see {@link heatmap.spaceY}) + * Amount of vertical space (in pixels) avaible to render the boxwhisker in + * (see {@link boxwhisker.spaceY}) * @param {number} [spaceY=undefined] - * @memberof heatmap# + * @memberof boxwhisker# * @property */ spaceY, - /** - * The internal key of the cell specifiying to which x axis key it belongs - * (see {@link heatmap.xKey}) - * @param {string} [xKey='x'] - * @memberof heatmap# + * Whether or not to allow boxwhisker to render elements pass the main spatial dimension + * given the orientation (see {@link boxwhisker#orient}), where {@link boxwhisker#orient}="horizontal" + * the main dimension is {@link boxwhisker#spaceX} and where {@link boxwhisker#orient}="vertical" + * the main dimension is {@link boxwhisker#spaceY} + * @param {boolean} [overflowQ=false] + * @memberof boxwhisker# * @property */ - xKey = 'x', + overflowQ = true, + /** - * The internal key of the cell specifiying to which y axis key it belongs - * (see {@link heatmap.yKey}) - * @param {string} [yKey='y'] - * @memberof heatmap# + * An array - putatively of other arrays - depicting how boxes should be arranged + * @param {Array[]} [grouping=undefined] + * @memberof boxwhisker# * @property */ - yKey = 'y', + grouping, + quartilesKey = 'quartiles', // key in object where quartiles are stored + quartilesKeys = ["0.00", "0.25", "0.50", "0.75", "1.00"], // keys in quartiles object mapping values to min, q1, q2, q3 and max + /** - * The internal key of the cell specifiying what value to use to determine the color - * (see {@link heatmap.vKey}) - * @param {string} [vKey='v'] - * @memberof heatmap# + * How to get the value of the boxwhisker + * @param {function} [valueExtractor=function(key, index) { return data[key][quartilesKey] }] + * @memberof boxwhisker# * @property */ - vKey = 'v', - + valueExtractor = function(key, index) { return data[key][quartilesKey] }, /** - * Function for extracting the the value from xKey. - * (see {@link heatmap.xExtractor}) - * @param {function} [xExtractor=function(key, i) { return data[key][xKey] }] - * @returns {string} - * @memberof heatmap# + * How to sort the boxes - if {@link boxwhisker#grouping} is not provided. + * @param {function} [sortingFunction=descending] + * @memberof boxwhisker# * @property + * default + * function(keyA, keyB) {return d3.descending( + * valueExtractor(keyA)[quartilesKeys[4]], + * valueExtractor(keyB)[quartilesKeys[4]] + * )} */ - xExtractor = function(key, i) {return data[key][xKey] }, + sortingFunction = function(keyA, keyB) {return d3.descending( + valueExtractor(keyA)[quartilesKeys[4]], + valueExtractor(keyB)[quartilesKeys[4]] + )}, /** - * Function for extracting the the value from yKey. - * (see {@link heatmap.yExtractor}) - * @param {function} [yExtractor=function(key, i) { return data[key][yKey] }] - * @returns {string} - * @memberof heatmap# + * The scale for which boxwhisker values should be transformed by + * @param {d3.scale} [scale=d3.scaleLinear] + * @memberof boxwhisker# * @property */ - yExtractor = function(key, i) { return data[key][yKey] }, - + scale = d3.scaleLinear(), /** - * Function for extracting the the value from vKey. - * (see {@link heatmap.vExtractor}) - * @param {function} [vExtractor=function(key, i) { return data[key][vKey] }] - * @returns {number} - * @memberof heatmap# + * The padding for the domain of the scale (see {@link boxwhisker#scale}) + * @param {number} [domainPadding=0.5] + * @memberof boxwhisker# * @property */ - vExtractor = function(key, i) { return data[key][vKey] }, - - + domainPadding = 0.5, /** - * Whether or not to allow heatmap to render elements pass the main spatial dimension - * given the orientation (see {@link heatmap#orient}), where {@link heatmap#orient}="bottom" or {@link heatmap#orient}="top" - * the main dimension is {@link heatmap#spaceX} and where {@link heatmap#orient}="left" or {@link heatmap#orient}="right" - * the main dimension is {@link heatmap#spaceY} - * @param {boolean} [overflowQ=false] - * @memberof heatmap# + * Default space for the spacer (percentage) of main dimension given the orientation + * (see {@link boxwhisker#orient}), where {@link boxwhisker#orient}="horizontal" + * the main dimension is {@link boxwhisker#spaceX} and where {@link boxwhisker#orient}="vertical" + * the main dimension is {@link boxwhisker#spaceY} between boxes + * @param {number} [objectSpacer=0.05] + * @memberof boxwhisker# * @property */ - overflowQ = false, - + objectSpacer = 0.05, /** - * Default space for the spacer (percentage) of main dimension given the orientation - * (see {@link heatmap#orient}), where {@link heatmap#orient}="horizontal" - * the main dimension is {@link heatmap#spaceX} and where {@link heatmap#orient}="vertical" - * the main dimension is {@link heatmap#spaceY} between bubbles - * @param {number} [objectSpacer=0.0] - * @memberof heatmap# + * The minimum size that an object can be + * @param {number} [minObjectSize=15] + * @memberof boxwhisker# * @property */ - objectSpacer = 0.0, + minObjectSize = 15, /** - * The minimum size that an object can be in the y dimension - * @param {number} [minObjectSize=50] - * @memberof heatmap# + * The maximum size that an object can be + * @param {number} [maxObjectSize=50] + * @memberof boxwhisker# * @property */ - yMinObjectSize = 50, + maxObjectSize = 50, /** - * The minimum size that an object can be in the x dimension - * @param {number} [minObjectSize=50] - * @memberof heatmap# + * Percent of box and whisker size that whiskers will be rendered + * see {@link boxwhisker#objectSize} + * @param {number} [whiskerWidthPercent=0.6] + * @memberof boxwhisker# * @property */ - xMinObjectSize = 50, + whiskerWidthPercent = .6, /** - * The maximum size that an object can be in the x dimension - * @param {number} [maxObjectSize=100] - * @memberof heatmap# + * Instance of ColorFunction + * @param {function} [colorFunction = colorFunction()] + * @memberof boxwhisker# * @property */ - xMaxObjectSize = 100, + colorFunction$$1 = colorFunction(), /** - * The maximum size that an object can be in the y dimension - * @param {number} [maxObjectSize=100] - * @memberof heatmap# + * The stroke width of the boxes + * @param {number} [boxStrokeWidth=2] + * @memberof boxwhisker# * @property */ - yMaxObjectSize = 100, - - + boxStrokeWidth = 2, /** - * The stroke width of the bubbles - * @param {number} [objectStrokeWidth=2] - * @memberof heatmap# + * The stroke width of the whiskers + * @param {number} [whiskerStrokeWidth=2] + * @memberof boxwhisker# * @property */ - objectStrokeWidth = 2, - // colorFunc = colorFunction(), + whiskerStrokeWidth = 2, /** * Color of the background * @param {string} [backgroundFill="transparent"] - * @memberof heatmap# + * @memberof boxwhisker# * @property */ backgroundFill = 'transparent', /** - * Namespace for all items made by this instance of heatmap - * @param {string} [namespace="d3sm-heatmap"] - * @memberof heatmap# + * Namespace for all items made by this instance of boxwhisker + * @param {string} [namespace="d3sm-box-whisker"] + * @memberof boxwhisker# * @property */ - namespace = 'd3sm-heatmap', + namespace = 'd3sm-box-whisker', /** - * Class name for heatmap container ( element) - * @param {string} [objectClass="heatmap"] - * @memberof heatmap# + * Class name for boxwhisker container ( element) + * @param {string} [objectClass="box-whisk"] + * @memberof boxwhisker# * @property */ - objectClass = 'heatmap', + objectClass = 'box-whisk', + /** * Duration of all transitions of this element * @param {number} [transitionDuration=1000] - * @memberof heatmap# + * @memberof boxwhisker# * @property */ transitionDuration = 1000, /** * Easing function for transitions * @param {d3.ease} [easeFunc=d3.easeExp] - * @memberof heatmap# + * @memberof boxwhisker# * @property */ easeFunc = d3.easeExp, /** - * Stores the keys of all the cells - * Calculated after heatmap called. - * @param {string[]} [cellKeys=undefined] - * @memberof heatmap# - * @property - */ - cellKeys, - /** - * Stores the list of utils.arr.unique xValues - * Calculated after heatmap called. - * @param {string[]} [xValues=undefined] - * @memberof heatmap# + * The keys of the boxes + * @param {string[]} [boxKeys=undefined] + * @memberof boxwhisker# * @property */ - xValues, + boxKeys, /** - * Stores the list of utils.arr.unique yValues - * Calculated after heatmap called. - * @param {string[]} [yValues=undefined] - * @memberof heatmap# + * The values of the boxes + * @param {string[]} [boxValues=undefined] + * @memberof boxwhisker# * @property */ - yValues, + boxValues, /** - * Stores the list of utils.arr.unique vValues - * Calculated after heatmap called. - * @param {string[]} [vValues=undefined] - * @memberof heatmap# + * The objectSize (actual width) used by the boxes + * @param {number} [objectSize=undefined] + * @memberof boxwhisker# * @property */ - vValues, - - xKeySortingFunction = function(a, b) { return xValues.indexOf(xExtractor(a)) - xValues.indexOf(xExtractor(b)) }, - yKeySortingFunction = function(a, b) { return yValues.indexOf(yExtractor(a)) - yValues.indexOf(yExtractor(b)) }, + objectSize, /** - * Instance of ColorFunction with .colorBy set to 'category' - * @function colorFunction - * @memberof heatmap# + * The spacerSize (actual width) used by the spacers between the boxes + * @param {number} [spacerSize=undefined] + * @memberof boxwhisker# * @property */ - colorFunction$$1 = colorFunction().colorBy('category'), + spacerSize, /** * Instance of Tooltip - * @function tooltip - * @memberof heatmap# - * @property - */ - tooltip$$1 = tooltip(), - - /** - * store the size the heatmap could be in the x dimension - * the actuall size of the heatmap will be the min of xSize and {@link heatmap#ySize} - * Calculated after heatmap called. - * @param {string[]} [xSize=undefined] - * @memberof heatmap# - * @property - */ - xSize, - /** - * store the size of the spacer in the x dimension - * Calculated after heatmap called. - * @param {string[]} [xSpacerSize=undefined] - * @memberof heatmap# - * @property - */ - xSpacerSize, - - /** - * store the size the heatmap could be in the y dimension - * the actuall size of the heatmap will be the min of xSize and {@link heatmap#xSize} - * Calculated after heatmap called. - * @param {string[]} [ySize=undefined] - * @memberof heatmap# - * @property - */ - ySize, - /** - * store the size of the spacer in the y dimension. - * Calculated after heatmap called. - * @param {string[]} [xSpacerSize=undefined] - * @memberof heatmap# + * @param {function} [tooltip=tooltip()] + * @memberof boxwhisker# * @property */ - ySpacerSize; - + tooltip$$1 = tooltip(); /** * Gets or sets the selection in which items are manipulated * @param {d3.selection} [_=none] - * @returns {heatmap | d3.selection} - * @memberof heatmap + * @returns {boxwhisker | d3.selection} + * @memberof boxwhisker * @property * by default selection = selection */ - hm.selection = function(_) { return arguments.length ? (selection = _, hm) : selection; }; + boxwhisker.selection = function(_) { return arguments.length ? (selection = _, boxwhisker) : selection; }; /** * Gets or sets the data - * (see {@link heatmap#data}) + * (see {@link boxwhisker#data}) * @param {number} [_=none] - * @returns {heatmap | object} - * @memberof heatmap + * @returns {boxwhisker | object} + * @memberof boxwhisker * @property */ - hm.data = function(_) { return arguments.length ? (data = _, hm) : data; }; - // hm.orient = function(_) { return arguments.length ? (orient = _, hm) : orient; } + boxwhisker.data = function(_) { return arguments.length ? (data = _, boxwhisker) : data; }; + /** + * Gets or sets the orient of the boxes + * (see {@link boxwhisker#orient}) + * @param {number} [_=none] + * @returns {boxwhisker | object} + * @memberof boxwhisker + * @property + */ + boxwhisker.orient = function(_) { return arguments.length ? (orient = _, boxwhisker) : orient; }; /** * Gets or sets the amount of horizontal space in which items are manipulated - * (see {@link heatmap#spaceX}) + * (see {@link boxwhisker#spaceX}) * @param {number} [_=none] should be a number > 0 - * @returns {heatmap | number} - * @memberof heatmap + * @returns {boxwhisker | number} + * @memberof boxwhisker * @property * by default spaceX = undefined */ - hm.spaceX = function(_) { return arguments.length ? (spaceX = _, hm) : spaceX; }; + boxwhisker.spaceX = function(_) { return arguments.length ? (spaceX = _, boxwhisker) : spaceX; }; /** * Gets or sets the amount of vertical space in which items are manipulated - * (see {@link heatmap#spaceY}) + * (see {@link boxwhisker#spaceY}) * @param {number} [_=none] should be a number > 0 - * @returns {heatmap | number} - * @memberof heatmap + * @returns {boxwhisker | number} + * @memberof boxwhisker * @property * by default spaceY = undefined */ - hm.spaceY = function(_) { return arguments.length ? (spaceY = _, hm) : spaceY; }; - + boxwhisker.spaceY = function(_) { return arguments.length ? (spaceY = _, boxwhisker) : spaceY; }; /** - * Gets or sets the xKey - * (see {@link heatmap#xKey}) - * @param {string} [_=none] - * @returns {heatmap | string} - * @memberof heatmap + * Gets / sets whether or not boxwhisker is allowed to go beyond specified dimensions + * (see {@link boxwhisker#overflowQ}) + * @param {boolean} [_=none] + * @returns {boxwhisker | boolean} + * @memberof boxwhisker * @property - * by default xKey = 'x' + * by default overflowQ = false */ - hm.xKey = function(_) { return arguments.length ? (xKey = _, hm) : xKey; }; + boxwhisker.overflowQ = function(_) { return arguments.length ? (overflowQ = _, boxwhisker) : overflowQ; }; /** - * Gets or sets the yKey - * (see {@link heatmap#yKey}) - * @param {string} [_=none] - * @returns {heatmap | string} - * @memberof heatmap + * Gets / sets the grouping of the boxes + * (see {@link boxwhisker#grouping}) + * @param {Array[]} [_=none] + * @returns {boxwhisker | Array[]} + * @memberof boxwhisker * @property - * by default yKey = 'y' + * by default grouping = undefined */ - hm.yKey = function(_) { return arguments.length ? (yKey = _, hm) : yKey; }; - + boxwhisker.grouping = function(_) { return arguments.length ? (grouping = _, boxwhisker) : grouping; }; /** - * Gets or sets the vKey - * (see {@link heatmap#vKey}) + * Gets / sets the quartilesKey + * (see {@link boxwhisker#quartilesKey}) * @param {string} [_=none] - * @returns {heatmap | string} - * @memberof heatmap + * @returns {boxwhisker | string} + * @memberof boxwhisker * @property - * by default vKey = 'y' + * by default quartilesKey = quartiles */ - hm.vKey = function(_) { return arguments.length ? (vKey = _, hm) : vKey; }; - + boxwhisker.quartilesKey = function(_) { return arguments.length ? (quartilesKey = _, boxwhisker) : quartilesKey; }; /** - * Gets or sets the cellKeys - * (see {@link heatmap#cellKeys}) + * Gets / sets the quartilesKeys + * (see {@link boxwhisker#quartilesKeys}) * @param {string[]} [_=none] - * @returns {heatmap | string[]} - * @memberof heatmap + * @returns {boxwhisker | string[]} + * @memberof boxwhisker * @property - * by default cellKeys = undefined + * by default quartilesKeys = [Q0, Q1, Q2, Q3, Q4] */ - hm.cellKeys = function(_) { return arguments.length ? (cellKeys = _, hm) : cellKeys; }; + boxwhisker.quartilesKeys = function(_) { return arguments.length ? (quartilesKeys = _, boxwhisker) : quartilesKeys; }; /** - * Gets or sets the xValues - * (see {@link heatmap#xValues}) - * @param {string[]} [_=none] - * @returns {heatmap | string[]} - * @memberof heatmap + * Gets / sets the valueExtractor + * (see {@link boxwhisker#valueExtractor}) + * @param {function} [_=none] + * @returns {boxwhisker | function} + * @memberof boxwhisker * @property - * by default xValues = undefined + * by default valueExtractor = function(key, index) { return data[key][quartilesKey] }, */ - hm.xValues = function(_) { return arguments.length ? (xValues = _, hm) : xValues; }; + + boxwhisker.valueExtractor = function(_) { return arguments.length ? (valueExtractor = _, boxwhisker) : valueExtractor; }; /** - * Gets or sets the yValues - * (see {@link heatmap#yValues}) - * @param {string[]} [_=none] - * @returns {heatmap | string[]} - * @memberof heatmap + * Gets / sets the sortingFunction + * (see {@link boxwhisker#sortingFunction}) + * @param {function} [_=none] + * @returns {boxwhisker | function} + * @memberof boxwhisker * @property - * by default yValues = undefined + * by default sortingFunction = function(keyA, keyB) {return d3.descending( + * valueExtractor(keyA)[quartilesKeys[4]], + * valueExtractor(keyB)[quartilesKeys[4]] + * )}, */ - hm.yValues = function(_) { return arguments.length ? (yValues = _, hm) : yValues; }; + boxwhisker.sortingFunction = function(_) { return arguments.length ? (sortingFunction = _, boxwhisker) : sortingFunction; }; /** - * Gets or sets the vValues - * (see {@link heatmap#vValues}) - * @param {number[]} [_=none] - * @returns {heatmap | number[]} - * @memberof heatmap + * Gets / sets the scale for which the boxwhisker values should be transformed by + * (see {@link boxwhisker#scale}) + * @param {d3.scale} [_=none] + * @returns {boxwhisker | d3.scale} + * @memberof boxwhisker * @property - * by default vValues = undefined + * by default scale = d3.scaleLinear() */ - hm.vValues = function(_) { return arguments.length ? (vValues = _, hm) : vValues; }; - - + boxwhisker.scale = function(_) { return arguments.length ? (scale = _, boxwhisker) : scale; }; /** - * Gets or sets the xExtractor - * (see {@link heatmap#xExtractor}) - * @param {function} [_=none] - * @returns {heatmap | function} - * @memberof heatmap + * Gets / sets the padding for the domain of the scale + * (see {@link boxwhisker#domainPadding}) + * @param {number} [_=none] + * @returns {boxwhisker | number} + * @memberof boxwhisker * @property - * by default xExtractor = undefined + * by default domainPadding = 0.5 */ - hm.xExtractor = function(_) { return arguments.length ? (xExtractor = _, hm) : xExtractor; }; + boxwhisker.domainPadding = function(_) { return arguments.length ? (domainPadding = _, boxwhisker) : domainPadding; }; /** - * Gets or sets the yExtractor - * (see {@link heatmap#yExtractor}) - * @param {function} [_=none] - * @returns {heatmap | function} - * @memberof heatmap + * Gets / sets objectSpacer + * (see {@link boxwhisker#objectSpacer}) + * @param {number} [_=none] + * @returns {boxwhisker | number} + * @memberof boxwhisker * @property - * by default yExtractor = undefined + * by default objectSpacer = 0.05 */ - hm.yExtractor = function(_) { return arguments.length ? (yExtractor = _, hm) : yExtractor; }; + boxwhisker.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, boxwhisker) : objectSpacer; }; /** - * Gets or sets the vExtractor - * (see {@link heatmap#vExtractor}) - * @param {function} [_=none] - * @returns {heatmap | function} - * @memberof heatmap + * Gets / sets the minObjectSize + * (see {@link boxwhisker#minObjectSize}) + * @param {number} [_=none] + * @returns {boxwhisker | number} + * @memberof boxwhisker * @property - * by default vExtractor = undefined + * by default minObjectSize = 15 */ - hm.vExtractor = function(_) { return arguments.length ? (vExtractor = _, hm) : vExtractor; }; - - /** - * Gets / sets whether or not heatmap is allowed to go beyond specified dimensions - * (see {@link heatmap#spaceX}) - * @param {boolean} [_=none] - * @returns {heatmap | boolean} - * @memberof heatmap - * @property - * by default overflowQ = false - */ - hm.overflowQ = function(_) { return arguments.length ? (overflowQ = _, hm) : overflowQ; }; - /** - * Gets / sets objectSpacer - * (see {@link heatmap#objectSpacer}) - * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap - * @property - * by default objectSpacer = 0.0 - */ - hm.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, objectSpacer) : data; }; + boxwhisker.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, boxwhisker) : minObjectSize; }; /** - * Gets / sets the minObjectSize - * (see {@link heatmap#minObjectSize}) + * Gets / sets the maxObjectSize + * (see {@link boxwhisker#maxObjectSize}) * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * @returns {boxwhisker | number} + * @memberof boxwhisker * @property - * by default minObjectSize = 50 + * by default maxObjectSize = 50 */ - hm.yMinObjectSize = function(_) { return arguments.length ? (yMinObjectSize = _, hm) : yMinObjectSize; }; + boxwhisker.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, boxwhisker) : maxObjectSize; }; /** - * Gets / sets the maxObjectSize - * (see {@link heatmap#maxObjectSize}) + * Gets / sets the whiskerWidthPercent + * (see {@link boxwhisker#whiskerWidthPercent}) * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * @returns {boxwhisker | number} + * @memberof boxwhisker * @property - * by default maxObjectSize = 100 + * by default maxObjectSize = 0.6 */ - hm.yMaxObjectSize = function(_) { return arguments.length ? (yMaxObjectSize = _, hm) : yMaxObjectSize; }; + boxwhisker.whiskerWidthPercent = function(_) { return arguments.length ? (whiskerWidthPercent = _, boxwhisker) : whiskerWidthPercent; }; /** - * Gets / sets the minObjectSize - * (see {@link heatmap#minObjectSize}) - * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * Gets / sets the colorFunction + * (see {@link boxwhisker#colorFunction}) + * @param {colorFunction} [_=none] + * @returns {boxwhisker | colorFunction} + * @memberof boxwhisker * @property - * by default minObjectSize = 50 + * by default colorFunction = colorFunction() */ - hm.xMinObjectSize = function(_) { return arguments.length ? (xMinObjectSize = _, hm) : xMinObjectSize; }; + boxwhisker.colorFunction = function(_) { return arguments.length ? (colorFunction$$1 = _, boxwhisker) : colorFunction$$1; }; /** - * Gets / sets the maxObjectSize - * (see {@link heatmap#maxObjectSize}) + * Gets / sets the boxStrokeWidth + * (see {@link boxwhisker#boxStrokeWidth}) * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * @returns {boxwhisker | number} + * @memberof boxwhisker * @property - * by default maxObjectSize = 100 + * by default boxStrokeWidth = 2 */ - hm.xMaxObjectSize = function(_) { return arguments.length ? (xMaxObjectSize = _, hm) : xMaxObjectSize; }; + boxwhisker.boxStrokeWidth = function(_) { return arguments.length ? (boxStrokeWidth = _, boxwhisker) : boxStrokeWidth; }; /** - * Gets / sets the objectStrokeWidth - * (see {@link heatmap#objectStrokeWidth}) + * Gets / sets the whiskerStrokeWidth + * (see {@link boxwhisker#whiskerStrokeWidth}) * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * @returns {boxwhisker | number} + * @memberof boxwhisker * @property - * by default objectStrokeWidth = 2 + * by default whiskerStrokeWidth = 2 */ - hm.objectStrokeWidth = function(_) { return arguments.length ? (objectStrokeWidth = _, hm) : objectStrokeWidth; }; + boxwhisker.whiskerStrokeWidth = function(_) { return arguments.length ? (whiskerStrokeWidth = _, boxwhisker) : whiskerStrokeWidth; }; + /** * Gets / sets the backgroundFill - * (see {@link heatmap#backgroundFill}) + * (see {@link boxwhisker#backgroundFill}) * @param {string} [_=none] - * @returns {heatmap | string} - * @memberof heatmap + * @returns {boxwhisker | string} + * @memberof boxwhisker * @property * by default backgroundFill = 'transparent' */ - hm.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, hm) : backgroundFill; }; + boxwhisker.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, boxwhisker) : backgroundFill; }; /** * Gets / sets the namespace - * (see {@link heatmap#namespace}) + * (see {@link boxwhisker#namespace}) * @param {string} [_=none] - * @returns {heatmap | string} - * @memberof heatmap + * @returns {boxwhisker | string} + * @memberof boxwhisker * @property - * by default namespace = 'd3sm-heatmap' + * by default namespace = 'd3sm-boxwhisker' */ - hm.namespace = function(_) { return arguments.length ? (namespace = _, hm) : namespace; }; + boxwhisker.namespace = function(_) { return arguments.length ? (namespace = _, boxwhisker) : namespace; }; /** * Gets / sets the objectClass - * (see {@link heatmap#objectClass}) + * (see {@link boxwhisker#objectClass}) * @param {string} [_=none] - * @returns {heatmap | string} - * @memberof heatmap + * @returns {boxwhisker | string} + * @memberof boxwhisker * @property * by default objectClass = 'tick-group' */ - hm.objectClass = function(_) { return arguments.length ? (objectClass = _, hm) : objectClass; }; + boxwhisker.objectClass = function(_) { return arguments.length ? (objectClass = _, boxwhisker) : objectClass; }; /** * Gets / sets the transitionDuration - * (see {@link heatmap#transitionDuration}) + * (see {@link boxwhisker#transitionDuration}) * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * @returns {boxwhisker | number} + * @memberof boxwhisker * @property * by default transitionDuration = 1000 */ - hm.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, hm) : transitionDuration; }; + boxwhisker.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, boxwhisker) : transitionDuration; }; /** * Gets / sets the easeFunc - * (see {@link heatmap#easeFunc}) + * (see {@link boxwhisker#easeFunc}) * @param {d3.ease} [_=none] - * @returns {heatmap | d3.ease} - * @memberof heatmap + * @returns {boxwhisker | d3.ease} + * @memberof boxwhisker * @property * by default easeFunc = d3.easeExp */ - hm.easeFunc = function(_) { return arguments.length ? (easeFunc = _, hm) : easeFunc; }; - - /** - * Gets / sets the tooltip - * (see {@link heatmap#tooltip}) - * @param {tooltip} [_=none] - * @returns {heatmap | tooltip} - * @memberof heatmap - * @property - * by default tooltip = tooltip() - */ - hm.tooltip = function(_) { return arguments.length ? (tooltip$$1 = _, hm) : tooltip$$1; }; + boxwhisker.easeFunc = function(_) { return arguments.length ? (easeFunc = _, boxwhisker) : easeFunc; }; /** - * Gets / sets the colorFunction - * (see {@link heatmap#colorFunction}) - * @param {colorFunction} [_=none] - * @returns {heatmap | colorFunction} - * @memberof heatmap + * Gets / sets the barKeys + * (see {@link boxwhisker#boxKeys}) + * @param {string[]} [_=none] + * @returns {boxwhisker | string[]} + * @memberof boxwhisker * @property - * by default colorFunction = colorFunction() + * by default boxKeys = undefined */ - hm.colorFunction = function(_) { return arguments.length ? (colorFunction$$1 = _, hm) : colorFunction$$1; }; - + boxwhisker.boxKeys = function(_) { return arguments.length ? (boxKeys = _, boxwhisker) : boxKeys; }; /** - * Gets / sets the xSize - * (see {@link heatmap#xSize}) - * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * Gets / sets the boxValues + * (see {@link boxwhisker#boxValues}) + * @param {number[]} [_=none] + * @returns {boxwhisker | number[]} + * @memberof boxwhisker * @property - * by default xSize = undefined + * by default boxValues = undefined */ - hm.xSize = function(_) { return arguments.length ? (xSize = _, hm) : xSize; }; + boxwhisker.boxValues = function(_) { return arguments.length ? (boxValues = _, boxwhisker) : boxValues; }; /** - * Gets / sets the xSpacerSize - * (see {@link heatmap#xSpacerSize}) + * Gets / sets the objectSize + * (see {@link boxwhisker#objectSize}) * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * @returns {boxwhisker | number} + * @memberof boxwhisker * @property - * by default xSpacerSize = undefined + * by default objectSize = undefined */ - hm.xSpacerSize = function(_) { return arguments.length ? (xSpacerSize = _, hm) : xSpacerSize; }; + boxwhisker.objectSize = function(_) { return arguments.length ? (objectSize = _, boxwhisker) : objectSize; }; /** - * Gets / sets the ySize - * (see {@link heatmap#ySize}) + * Gets / sets the spacerSize + * (see {@link boxwhisker#spacerSize}) * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * @returns {boxwhisker | number} + * @memberof boxwhisker * @property - * by default ySize = undefined + * by default spacerSize = undefined */ - hm.ySize = function(_) { return arguments.length ? (ySize = _, hm) : ySize; }; + boxwhisker.spacerSize = function(_) { return arguments.length ? (spacerSize = _, boxwhisker) : spacerSize; }; /** - * Gets / sets the ySpacerSize - * (see {@link heatmap#ySpacerSize}) - * @param {number} [_=none] - * @returns {heatmap | number} - * @memberof heatmap + * Gets / sets the tooltip + * (see {@link boxwhisker#tooltip}) + * @param {tooltip} [_=none] + * @returns {boxwhisker | tooltip} + * @memberof boxwhisker * @property - * by default ySpacerSize = undefined + * by default tooltip = tooltip() */ - hm.ySpacerSize = function(_) { return arguments.length ? (ySpacerSize = _, hm) : ySpacerSize; }; - // hm.yKeySortingFunction = function(_) { return arguments.length ? (yKeySortingFunction = _, hm) : yKeySortingFunction; } - // hm.xKeySortingFunction = function(_) { return arguments.length ? (xKeySortingFunction = _, hm) : xKeySortingFunction; } + boxwhisker.tooltip = function(_) { return arguments.length ? (tooltip$$1 = _, boxwhisker) : tooltip$$1; }; + function boxwhisker() { + // for convenience in handling orientation specific values + var horizontalQ = (orient == 'horizontal') ? true : false; + var verticalQ = !horizontalQ; - function hm() { + // background cliping rectangle var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; - var container = utils$1.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); - - cellKeys = d3.keys(data); + var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); - xValues = utils$1.arr.unique(cellKeys.map(xExtractor)); - yValues = utils$1.arr.unique(cellKeys.map(yExtractor)); - vValues = utils$1.arr.unique(cellKeys.map(vExtractor)); + // if grouping is undefined sort keys by sorting funct + var ordered = (grouping == undefined) ? d3.keys(data).sort(sortingFunction) : grouping; + // to prevent re-calculation and getters to be passed to axes + boxKeys = utils.arr.flatten(ordered); + boxValues = boxKeys.map(valueExtractor); - cellKeys.sort(function(a, b){ return xKeySortingFunction(a, b) || yKeySortingFunction(a, b) }); - utils$1.con.log('heatmap', 'cells are sorted by', cellKeys); + var numberOfObjects = boxKeys.length; + var extent = [ + Math.min(...boxValues.map(function(d,i){return d[quartilesKeys[0]]})) - domainPadding, + Math.max(...boxValues.map(function(d,i){return d[quartilesKeys[4]]})) + domainPadding + ]; + // set the scale + scale.domain(extent).range(horizontalQ ? [0,spaceY] : [spaceX, 0]); + var space = horizontalQ ? spaceX : spaceY; + // calculate object size + objectSize = utils.math.calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ); + // calculate spacer size if needed + spacerSize = utils.math.calculateWidthOfSpacer(boxKeys, space, objectSize, numberOfObjects, objectSpacer, overflowQ); + // make the nested groups + var spacerFunction = groupingSpacer() + .horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects) + .objectClass(objectClass).objectSize(objectSize).spacerSize(spacerSize) + .transitionDuration(transitionDuration).easeFunc(easeFunc) + .namespace(namespace); - utils$1.con.log('heatmap', 'x and y keys are', {x: xValues, y:yValues}); + // move stuff + spacerFunction(container, ordered, 0); + var parentIndexArray = []; + container.selectAll('g:not(.to-remove).'+objectClass) + .each(function(d, i){if (utils.arr.hasQ(boxKeys, d)){ parentIndexArray.push(Number(d3.select(this).attr('parent-index')));}}); - var xDim = xValues.length, yDim = yValues.length; - ySize = utils$1.math.calculateWidthOfObject(spaceY, yDim, yMinObjectSize, yMaxObjectSize, objectSpacer, overflowQ); - xSize = utils$1.math.calculateWidthOfObject(spaceX, xDim, xMinObjectSize, xMaxObjectSize, objectSpacer, overflowQ); - ySpacerSize = utils$1.math.calculateWidthOfSpacer(yValues, spaceY, ySize, yDim, objectSpacer, overflowQ); - xSpacerSize = utils$1.math.calculateWidthOfSpacer(xValues, spaceX, xSize, xDim, objectSpacer, overflowQ); - // console.table({ - // x:{ - // object: xSize, - // spacer: xSpacerSize, - // dim: xDim - // }, - // y:{ - // object: ySize, - // spacer: ySpacerSize, - // dim: yDim - // } - // - // }) - utils$1.con.log('heatmap', 'size of', {x: xSize, y: ySize}); + colorFunction$$1 = colorFunction$$1.colorBy() == 'index' + ? colorFunction$$1.dataExtent([0, Math.max(...parentIndexArray)]) + : colorFunction$$1.dataExtent(extent); + // set attributes for box and whiskers + container.selectAll('g:not(.to-remove).'+objectClass).each(function(key, i) { + var t = d3.select(this), + currentData = data[key], - var ySpacer = groupingSpacer() - .horizontalQ(false) - .moveby('category') - .numberOfObjects(yDim) - .objectClass(utils$1.str.hypenate(objectClass, 'row')) - .objectSize(ySize + ySpacerSize) - .spacerSize(0) - .transitionDuration(transitionDuration) - .easeFunc(easeFunc) - .namespace('row'); + quartiles$$1 = valueExtractor(key, i), + q0 = quartiles$$1[quartilesKeys[0]], + q1 = quartiles$$1[quartilesKeys[1]], + q2 = quartiles$$1[quartilesKeys[2]], + q3 = quartiles$$1[quartilesKeys[3]], + q4 = quartiles$$1[quartilesKeys[4]]; - var xSpacer = groupingSpacer() - .horizontalQ(true) - .moveby('category') - .numberOfObjects(xDim) - .objectClass(objectClass) - .objectSize(xSize + xSpacerSize) - .spacerSize(0) - .transitionDuration(transitionDuration) - .easeFunc(easeFunc); + var i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'), + fillColor = colorFunction$$1(key, q2, i, 'fill'), // prevent duplicate computation + strokeColor = colorFunction$$1(key, q2, i, 'stroke'); - ySpacer(container, yValues, 0); - container.selectAll('g.'+utils$1.str.hypenate(objectClass, 'row')) - .each(function(d, i){ xSpacer(d3.select(this), xValues, 0); }); + var + whisk = utils.sel.safeSelect(t, 'g', 'whisker'), + uWhisk = utils.sel.safeSelect(whisk, 'path', 'upper'), + lWhisk = utils.sel.safeSelect(whisk, 'path', 'lower'), + quart = utils.sel.safeSelect(t, 'g', 'quartile'), + uQuart = utils.sel.safeSelect(quart, 'rect', 'upper'), + lQuart = utils.sel.safeSelect(quart, 'rect', 'lower'), + mQuart = utils.sel.safeSelect(quart, 'circle', 'median'); + + + // set upper quartile (q3) + uQuart.transition().duration(transitionDuration).ease(easeFunc) + .attr('width', horizontalQ ? objectSize : scale(q3) - scale(q2)) + .attr('height', verticalQ ? objectSize : scale(q3) - scale(q2)) + .attr('fill', fillColor) + .attr('stroke', strokeColor) + .attr('stroke-width', boxStrokeWidth) + .attr('transform', function(d, i){ + var + x = horizontalQ ? 0 : scale(q2), + y = verticalQ ? 0 : scale(extent[1]) - scale(q3), + t = 'translate('+x+','+y+')'; + return t + }); - var cells = container.selectAll('g:not(.to-remove).'+objectClass); + // set lower quartile (q1) + lQuart.transition().duration(transitionDuration).ease(easeFunc) + .attr('width', horizontalQ ? objectSize : scale(q2) - scale(q1)) + .attr('height', verticalQ ? objectSize : scale(q2) - scale(q1)) + .attr('fill', fillColor) + .attr('stroke', strokeColor) + .attr('stroke-width', boxStrokeWidth) + .attr('transform', function(d, i){ + var + x = horizontalQ ? 0 : scale(q1), + y = verticalQ ? 0 : scale(extent[1]) - scale(q2), + t = 'translate('+x+','+y+')'; + return t + }); - if (cellKeys.length != yValues.length * xValues.length) { - var lookup = {}; - cellKeys.map(function(k, i){ - lookup[xExtractor(k)+'::'+yExtractor(k)] = k; + // set median (q2) + mQuart.transition().duration(transitionDuration).ease(easeFunc) + .attr('r', function(d, i){ + var r = objectSize / 2; + var dif = (scale(q3) - scale(q1)) / 2; + return (r > dif) ? dif : r + }) + .attr('fill', fillColor) + .attr('stroke', strokeColor) + .attr('stroke-width', boxStrokeWidth) + .attr('transform', function(d, i){ + var + x = horizontalQ ? objectSize / 2 : scale(q2), + y = verticalQ ? objectSize / 2 : scale(extent[1]) - scale(q2), + t = 'translate('+x+','+y+')'; + return t }); - var positionedCellKeys = []; - for (var i = 0; i < yValues.length; i++) { - for (var j = 0; j < xValues.length; j++) { - var lookupValue = lookup[xValues[j]+"::"+yValues[i]]; - if (lookupValue == undefined) { - positionedCellKeys.push(undefined); - } else { - positionedCellKeys.push(lookupValue); - } - } - } + // set lower whisker (min) + lWhisk.transition().duration(transitionDuration).ease(easeFunc) + .attr('d', function(dd, ii){ + var + dir = false, + x = 0, + y = 0, + h = horizontalQ ? scale(q1) - scale(q0) : objectSize, + w = verticalQ ? scale(q1) - scale(q0) : objectSize; + return utils.paths.whiskerPath(dir, x, y, w, h, whiskerWidthPercent, orient) + }) + .attr('transform', function(d, i){ + var + x = horizontalQ ? 0 : scale(q1), + y = verticalQ ? 0 : scale(extent[1]) - scale(q1), + t = 'translate('+x+','+y+')'; + return t + }) + .attr('stroke', 'black').attr('stroke-width', whiskerStrokeWidth) + .attr('fill', 'none'); - cells.data(positionedCellKeys); - - // maybe breaks this - // !!!!!! IMPORTANT NOTE TODO LOOK HERE BUG - cellKeys = positionedCellKeys; - } else { - cells.data(cellKeys); - } - - - - var parentIndexArray = []; - cells.each(function(d, i){ parentIndexArray.push(Number(d3.select(this).attr('parent-index'))); }); - - colorFunction$$1 = colorFunction$$1.colorBy() == 'index' - ? colorFunction$$1.dataExtent([0, Math.max(...parentIndexArray)]) - : colorFunction$$1.dataExtent([0, Math.max(...vValues)]); - - cells.each(function(key, i) { - utils$1.con.log('heatmap', 'each cell', {key: key, index: i, node: d3.select(this).node()}); - - var t = d3.select(this); - if (key == undefined) {return} - var currentData = data[key], - value = vExtractor(key, i), - i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'), - fillColor = colorFunction$$1(key, value, i, 'fill'), // prevent duplicate computation - strokeColor = colorFunction$$1(key, value, i, 'stroke'); - - var c = utils$1.sel.safeSelect(t, 'rect', utils$1.str.hypenate(objectClass,'rect')); - c.attr('width', xSize + xSpacerSize - objectStrokeWidth) - .attr('height', ySize + ySpacerSize - objectStrokeWidth) - .attr('fill', fillColor) - .attr('x', objectStrokeWidth/2) - .attr('y', objectStrokeWidth/2) - .attr('stroke', "#000") - .attr('stroke-width', objectStrokeWidth); + // set upper whisker (max) + uWhisk.transition().duration(transitionDuration).ease(easeFunc) + .attr('d', function(dd, ii){ + var + dir = true, + x = 0, + y = 0, + h = horizontalQ ? scale(q4) - scale(q3) : objectSize, + w = verticalQ ? scale(q4) - scale(q3) : objectSize; + return utils.paths.whiskerPath(dir, x, y, w, h, whiskerWidthPercent, orient) + }) + .attr('transform', function(d, i){ + var + x = horizontalQ ? 0 : scale(q3), + y = verticalQ ? 0 : scale(extent[1]) - scale(q4), + t = 'translate('+x+','+y+')'; + return t + }) + .attr('stroke', 'black') + .attr('stroke-width', whiskerStrokeWidth) + .attr('fill', 'none'); }); - tooltip$$1.selection(cells.selectAll('rect.'+utils$1.str.hypenate(objectClass, 'rect'))) + tooltip$$1.selection(container.selectAll('g:not(.to-remove).'+objectClass)) .data(data); - // .keys(['r', 'v']) - // .header(function(d, i){return utils.str.hypenate(data[d][xKey], data[d][yKey]) }) - tooltip$$1(); } - return hm; + return boxwhisker } - /******************************************************************************* - ** ** - ** ** - ** VIOLIN ** - ** ** - ** ** - *******************************************************************************/ - /** - * Creates a violin + * Creates a heatmap * - * {@link https://sumneuron.gitlab.io/d3sm/demos/basic-violins/index.html Demo} - * @constructor violin + * {@link https://sumneuron.gitlab.io/d3sm/demos/heatmap-heatmap/index.html Demo} + * @constructor heatmap * @param {d3.selection} selection - * @namespace violin - * @returns {function} violin + * @namespace heatmap + * @returns {function} heatmap */ - function violin( selection ) { + function heatmap( selection ) { var /** - * Data to plot. Assumed to be a object, where each key corresponds to a violin - * (see {@link violin#data}) + * Data to plot. Assumed to be a object, where each key corresponds to a cell + * (see {@link heatmap#data}) * @param {Object} [data=undefined] - * @memberof violin# + * @memberof heatmap# * @property */ data, + /** - * Which direction to render the bars in - * (see {@link violin#orient}) - * @param {number} [orient='horizontal'] - * @memberof violin# - * @property - */ - orient='horizontal', - /** - * Amount of horizontal space (in pixels) avaible to render the violin in - * (see {@link violin#spaceX}) + * Amount of horizontal space (in pixels) avaible to render the heatmap in + * (see {@link heatmap#spaceX}) * @param {number} [spaceX=undefined] - * @memberof violin# + * @memberof heatmap# * @property */ spaceX, /** - * Amount of vertical space (in pixels) avaible to render the violin in - * (see {@link violin.spaceY}) + * Amount of vertical space (in pixels) avaible to render the heatmap in + * (see {@link heatmap.spaceY}) * @param {number} [spaceY=undefined] - * @memberof violin# + * @memberof heatmap# * @property */ spaceY, + /** - * Whether or not to allow violin to render elements pass the main spatial dimension - * given the orientation (see {@link violin#orient}), where {@link violin#orient}="horizontal" - * the main dimension is {@link violin#spaceX} and where {@link violin#orient}="vertical" - * the main dimension is {@link violin#spaceY} - * @param {boolean} [overflowQ=false] - * @memberof violin# + * The internal key of the cell specifiying to which x axis key it belongs + * (see {@link heatmap.xKey}) + * @param {string} [xKey='x'] + * @memberof heatmap# * @property */ - overflowQ = true, + xKey = 'x', /** - * Whether or not to display points inside the points - * @param {boolean} [pointsQ=false] - * @memberof violin# + * The internal key of the cell specifiying to which y axis key it belongs + * (see {@link heatmap.yKey}) + * @param {string} [yKey='y'] + * @memberof heatmap# * @property */ - pointsQ = true, + yKey = 'y', + /** - * An array - putatively of other arrays - depicting how bars should be arranged - * @param {Array[]} [grouping=undefined] - * @memberof violin# + * The internal key of the cell specifiying what value to use to determine the color + * (see {@link heatmap.vKey}) + * @param {string} [vKey='v'] + * @memberof heatmap# * @property */ - grouping, + vKey = 'v', + /** - * How to get the value of the violin - * @param {function} [valueExtractor=function(key, index) { return data[key] }] - * @memberof violin# + * Function for extracting the the value from xKey. + * (see {@link heatmap.xExtractor}) + * @param {function} [xExtractor=function(key, i) { return data[key][xKey] }] + * @returns {string} + * @memberof heatmap# * @property */ - valueExtractor = function(key, index) {return data[key] }, + xExtractor = function(key, i) {return data[key][xKey] }, /** - * How to sort the bars - if {@link violin#grouping} is not provided. - * @param {function} [sortingFunction=function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}] - * @memberof violin# + * Function for extracting the the value from yKey. + * (see {@link heatmap.yExtractor}) + * @param {function} [yExtractor=function(key, i) { return data[key][yKey] }] + * @returns {string} + * @memberof heatmap# * @property */ - sortingFunction = function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}, + yExtractor = function(key, i) { return data[key][yKey] }, /** - * The scale for which violin values should be transformed by - * @param {d3.scale} [scale=d3.scaleLinear] - * @memberof violin# + * Function for extracting the the value from vKey. + * (see {@link heatmap.vExtractor}) + * @param {function} [vExtractor=function(key, i) { return data[key][vKey] }] + * @returns {number} + * @memberof heatmap# * @property */ - scale = d3.scaleLinear(), + vExtractor = function(key, i) { return data[key][vKey] }, + + /** - * The padding for the domain of the scale (see {@link violin#scale}) - * @param {number} [domainPadding=0.5] - * @memberof violin# + * Whether or not to allow heatmap to render elements pass the main spatial dimension + * given the orientation (see {@link heatmap#orient}), where {@link heatmap#orient}="bottom" or {@link heatmap#orient}="top" + * the main dimension is {@link heatmap#spaceX} and where {@link heatmap#orient}="left" or {@link heatmap#orient}="right" + * the main dimension is {@link heatmap#spaceY} + * @param {boolean} [overflowQ=false] + * @memberof heatmap# * @property */ - domainPadding = 0.5, + overflowQ = false, + /** * Default space for the spacer (percentage) of main dimension given the orientation - * (see {@link violin#orient}), where {@link violin#orient}="horizontal" - * the main dimension is {@link violin#spaceX} and where {@link violin#orient}="vertical" - * the main dimension is {@link violin#spaceY} between bars - * @param {number} [objectSpacer=0.05] - * @memberof violin# + * (see {@link heatmap#orient}), where {@link heatmap#orient}="horizontal" + * the main dimension is {@link heatmap#spaceX} and where {@link heatmap#orient}="vertical" + * the main dimension is {@link heatmap#spaceY} between bubbles + * @param {number} [objectSpacer=0.0] + * @memberof heatmap# * @property */ - objectSpacer = 0.05, + objectSpacer = 0.0, /** - * The minimum size that an object can be + * The minimum size that an object can be in the y dimension * @param {number} [minObjectSize=50] - * @memberof violin# - * @property - */ - minObjectSize = 50, - /** - * The maximum size that an object can be - * @param {number} [maxObjectSize=100] - * @memberof violin# + * @memberof heatmap# * @property */ - maxObjectSize = 100, - + yMinObjectSize = 50, /** - * The stroke width of the bars - * @param {number} [barStrokeWidth=2] - * @memberof violin# + * The minimum size that an object can be in the x dimension + * @param {number} [minObjectSize=50] + * @memberof heatmap# * @property */ - objectStrokeWidth = 2, + xMinObjectSize = 50, /** - * Instance of ColorFunction - * @param {function} [colorFunction = colorFunction()] - * @memberof violin# + * The maximum size that an object can be in the x dimension + * @param {number} [maxObjectSize=100] + * @memberof heatmap# * @property */ - colorFunction$$1 = colorFunction(), + xMaxObjectSize = 100, /** - * Instance of ColorFunction modified by a scale for the points - * @param {function} [pointColorFunc = colorFunction()] - * @memberof violin# + * The maximum size that an object can be in the y dimension + * @param {number} [maxObjectSize=100] + * @memberof heatmap# * @property */ - pointColorFunc = function (d, type, base, min, max) { - var minMaxHexScale = d3.scaleLinear().domain([min, max]).range([-0.25, 0.05]); - var scaledColor = utils.color.modifyHexidecimalColorLuminance(base.replace('#', ''), minMaxHexScale(d)); - var mod = type == "stroke" ? 0 : 0.25; - return utils.color.modifyHexidecimalColorLuminance(scaledColor.replace('#', ''), mod) - }, + yMaxObjectSize = 100, + /** - * The radius of a point - * @param {number} [pointRadius=3] - * @memberof violin# - * @property - */ - pointRadius = 3, - /** - * The stroke width of the oints - * @param {number} [pointStrokeWidth=2] - * @memberof violin# + * The stroke width of the bubbles + * @param {number} [objectStrokeWidth=2] + * @memberof heatmap# * @property */ - pointStrokeWidth = 2, + objectStrokeWidth = 2, + // colorFunc = colorFunction(), /** * Color of the background * @param {string} [backgroundFill="transparent"] - * @memberof violin# + * @memberof heatmap# * @property */ backgroundFill = 'transparent', /** - * Namespace for all items made by this instance of violin - * @param {string} [namespace="d3sm-violin"] - * @memberof violin# + * Namespace for all items made by this instance of heatmap + * @param {string} [namespace="d3sm-heatmap"] + * @memberof heatmap# * @property */ - namespace = 'd3sm-violin', + namespace = 'd3sm-heatmap', /** - * Class name for violin container ( element) - * @param {string} [objectClass="violin"] - * @memberof violin# + * Class name for heatmap container ( element) + * @param {string} [objectClass="heatmap"] + * @memberof heatmap# * @property */ - objectClass = 'violin', + objectClass = 'heatmap', /** * Duration of all transitions of this element * @param {number} [transitionDuration=1000] - * @memberof violin# + * @memberof heatmap# * @property */ transitionDuration = 1000, /** * Easing function for transitions * @param {d3.ease} [easeFunc=d3.easeExp] - * @memberof violin# + * @memberof heatmap# * @property */ easeFunc = d3.easeExp, /** - * The keys corresponding to each quartile - * @param {string[]} [quartileKeys=["Q0", "Q1", "Q2", "Q3", "Q4"]] - * @memberof violin# + * Stores the keys of all the cells + * Calculated after heatmap called. + * @param {string[]} [cellKeys=undefined] + * @memberof heatmap# * @property */ - quartileKeys = ["Q0", "Q1", "Q2", "Q3", "Q4"], - + cellKeys, /** - * The keys of the bars - * @param {string[]} [violinKeys=undefined] - * @memberof violin# + * Stores the list of utils.arr.unique xValues + * Calculated after heatmap called. + * @param {string[]} [xValues=undefined] + * @memberof heatmap# * @property */ - violinKeys, + xValues, /** - * The values of the bars - * @param {number[]} [violinValues=undefined] - * @memberof violin# + * Stores the list of utils.arr.unique yValues + * Calculated after heatmap called. + * @param {string[]} [yValues=undefined] + * @memberof heatmap# * @property */ - violinValues, + yValues, /** - * The objectSize (actual width) used by the bars - * @param {number} [objectSize=undefined] - * @memberof violin# + * Stores the list of utils.arr.unique vValues + * Calculated after heatmap called. + * @param {string[]} [vValues=undefined] + * @memberof heatmap# * @property */ - objectSize, + vValues, + + xKeySortingFunction = function(a, b) { return xValues.indexOf(xExtractor(a)) - xValues.indexOf(xExtractor(b)) }, + yKeySortingFunction = function(a, b) { return yValues.indexOf(yExtractor(a)) - yValues.indexOf(yExtractor(b)) }, /** - * The spacerSize (actual width) used by the spacers between the bars - * @param {number} [spacerSize=undefined] - * @memberof violin# + * Instance of ColorFunction with .colorBy set to 'category' + * @function colorFunction + * @memberof heatmap# * @property */ - spacerSize, - + colorFunction$$1 = colorFunction().colorBy('category'), /** * Instance of Tooltip - * @param {function} [tooltip=tooltip()] - * @memberof violin# + * @function tooltip + * @memberof heatmap# * @property */ - tooltip$$1 = tooltip().keys([quartileKeys[4], quartileKeys[3], quartileKeys[2], quartileKeys[1], quartileKeys[0]]), - pointsTooltip = tooltip(), - // pointKeyExtractor = function(violinKey, violinData, violinValues) {return d3.keys(violinValues[violinKey].values)}, - // pointValueExtractor = function(pointKey, violinKey, violinData, violinValues) {return violinValues[violinKey].values[pointKey]}, - + tooltip$$1 = tooltip(), /** - * Function which given the key of the violin and that key's associated value - * returns the object consiting of the points of the violin - * (see {@link violin#violinPointsExtractor}) - * @param {Object} [violinPointsExtractor=function(violinKey, violinData) { return violinData.points }] - * @memberof violin# + * store the size the heatmap could be in the x dimension + * the actuall size of the heatmap will be the min of xSize and {@link heatmap#ySize} + * Calculated after heatmap called. + * @param {string[]} [xSize=undefined] + * @memberof heatmap# * @property */ - violinPointsExtractor = function (violinKey, violinData) {return violinData.points }, + xSize, /** - * Function which given the key of the current point and the object of points for the - * violin, returns the numerical value of the point - * (see {@link violin#violinPointValueExtractor}) - * @param {Object} [violinPointValueExtractor=function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value }] - * @memberof violin# + * store the size of the spacer in the x dimension + * Calculated after heatmap called. + * @param {string[]} [xSpacerSize=undefined] + * @memberof heatmap# * @property */ - violinPointValueExtractor = function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value }; - + xSpacerSize, /** - * Gets or sets the violinPointsExtractor - * @param {function} [_=none] - * @returns {violin | function} - * @memberof violin - * @property - * by default violinPointsExtractor = function(violinKey, violinData) { return violinData.points } - */ - violin.violinPointsExtractor = function(_) { return arguments.length ? (violinPointsExtractor = _, violin) : violinPointsExtractor; }; + * store the size the heatmap could be in the y dimension + * the actuall size of the heatmap will be the min of xSize and {@link heatmap#xSize} + * Calculated after heatmap called. + * @param {string[]} [ySize=undefined] + * @memberof heatmap# + * @property + */ + ySize, /** - * Gets or sets the violinPointValueExtractor - * @param {function} [_=none] - * @returns {violin | function} - * @memberof violin - * @property - * by default violinPointsExtractor = function(pointKey, violinKey, violinData, violinValues) {return violinValues[violinKey].values[pointKey]} - */ - violin.violinPointValueExtractor = function(_) { return arguments.length ? (violinPointValueExtractor = _, violin) : violinPointValueExtractor; }; - + * store the size of the spacer in the y dimension. + * Calculated after heatmap called. + * @param {string[]} [xSpacerSize=undefined] + * @memberof heatmap# + * @property + */ + ySpacerSize; /** * Gets or sets the selection in which items are manipulated * @param {d3.selection} [_=none] - * @returns {violin | d3.selection} - * @memberof violin + * @returns {heatmap | d3.selection} + * @memberof heatmap * @property * by default selection = selection */ - violin.selection = function(_) { return arguments.length ? (selection = _, violin) : selection; }; + hm.selection = function(_) { return arguments.length ? (selection = _, hm) : selection; }; /** * Gets or sets the data - * (see {@link violin#data}) - * @param {number} [_=none] - * @returns {violin | object} - * @memberof violin - * @property - */ - violin.data = function(_) { return arguments.length ? (data = _, violin) : data; }; - /** - * Gets or sets the orient of the boxes - * (see {@link violin#orient}) + * (see {@link heatmap#data}) * @param {number} [_=none] - * @returns {violin | object} - * @memberof violin + * @returns {heatmap | object} + * @memberof heatmap * @property */ - violin.orient = function(_) { return arguments.length ? (orient = _, violin) : orient; }; + hm.data = function(_) { return arguments.length ? (data = _, hm) : data; }; + // hm.orient = function(_) { return arguments.length ? (orient = _, hm) : orient; } /** * Gets or sets the amount of horizontal space in which items are manipulated - * (see {@link violin#spaceX}) + * (see {@link heatmap#spaceX}) * @param {number} [_=none] should be a number > 0 - * @returns {violin | number} - * @memberof violin + * @returns {heatmap | number} + * @memberof heatmap * @property * by default spaceX = undefined */ - violin.spaceX = function(_) { return arguments.length ? (spaceX = _, violin) : spaceX; }; + hm.spaceX = function(_) { return arguments.length ? (spaceX = _, hm) : spaceX; }; /** * Gets or sets the amount of vertical space in which items are manipulated - * (see {@link violin#spaceY}) + * (see {@link heatmap#spaceY}) * @param {number} [_=none] should be a number > 0 - * @returns {violin | number} - * @memberof violin + * @returns {heatmap | number} + * @memberof heatmap * @property * by default spaceY = undefined */ - violin.spaceY = function(_) { return arguments.length ? (spaceY = _, violin) : spaceY; }; - + hm.spaceY = function(_) { return arguments.length ? (spaceY = _, hm) : spaceY; }; /** - * Gets / sets whether or not violin is allowed to go beyond specified dimensions - * (see {@link violin#overflowQ}) - * @param {boolean} [_=none] - * @returns {violin | boolean} - * @memberof violin + * Gets or sets the xKey + * (see {@link heatmap#xKey}) + * @param {string} [_=none] + * @returns {heatmap | string} + * @memberof heatmap * @property - * by default overflowQ = false + * by default xKey = 'x' */ - violin.overflowQ = function(_) { return arguments.length ? (overflowQ = _, violin) : overflowQ; }; + hm.xKey = function(_) { return arguments.length ? (xKey = _, hm) : xKey; }; /** - * Gets / sets whether or not to plot points with the violins - * (see {@link violin#pointsQ}) - * @param {boolean} [_=none] - * @returns {violin | boolean} - * @memberof violin + * Gets or sets the yKey + * (see {@link heatmap#yKey}) + * @param {string} [_=none] + * @returns {heatmap | string} + * @memberof heatmap * @property - * by default pointsQ = false + * by default yKey = 'y' */ - violin.pointsQ = function(_) { return arguments.length ? (pointsQ = _, violin) : pointsQ; }; - + hm.yKey = function(_) { return arguments.length ? (yKey = _, hm) : yKey; }; /** - * Gets / sets the grouping of the boxes - * (see {@link violin#grouping}) - * @param {Array[]} [_=none] - * @returns {violin | Array[]} - * @memberof violin + * Gets or sets the vKey + * (see {@link heatmap#vKey}) + * @param {string} [_=none] + * @returns {heatmap | string} + * @memberof heatmap * @property - * by default grouping = undefined + * by default vKey = 'y' */ - violin.grouping = function(_) { return arguments.length ? (grouping = _, violin) : grouping; }; + hm.vKey = function(_) { return arguments.length ? (vKey = _, hm) : vKey; }; + /** - * Gets / sets the valueExtractor - * (see {@link violin#valueExtractor}) - * @param {function} [_=none] - * @returns {violin | function} - * @memberof violin + * Gets or sets the cellKeys + * (see {@link heatmap#cellKeys}) + * @param {string[]} [_=none] + * @returns {heatmap | string[]} + * @memberof heatmap * @property + * by default cellKeys = undefined */ - violin.valueExtractor = function(_) { return arguments.length ? (valueExtractor = _, violin) : valueExtractor; }; + hm.cellKeys = function(_) { return arguments.length ? (cellKeys = _, hm) : cellKeys; }; /** - * Gets / sets the sortingFunction - * (see {@link violin#sortingFunction}) - * @param {function} [_=none] - * @returns {violin | function} - * @memberof violin + * Gets or sets the xValues + * (see {@link heatmap#xValues}) + * @param {string[]} [_=none] + * @returns {heatmap | string[]} + * @memberof heatmap * @property + * by default xValues = undefined */ - violin.sortingFunction = function(_) { return arguments.length ? (sortingFunction = _, violin) : sortingFunction; }; - + hm.xValues = function(_) { return arguments.length ? (xValues = _, hm) : xValues; }; /** - * Gets / sets the scale for which the violin values should be transformed by - * (see {@link violin#scale}) - * @param {d3.scale} [_=none] - * @returns {violin | d3.scale} - * @memberof violin + * Gets or sets the yValues + * (see {@link heatmap#yValues}) + * @param {string[]} [_=none] + * @returns {heatmap | string[]} + * @memberof heatmap * @property - * by default scale = d3.scaleLinear() + * by default yValues = undefined */ - violin.scale = function(_) { return arguments.length ? (scale = _, violin) : scale; }; + hm.yValues = function(_) { return arguments.length ? (yValues = _, hm) : yValues; }; /** - * Gets / sets the padding for the domain of the scale - * (see {@link violin#domainPadding}) - * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin + * Gets or sets the vValues + * (see {@link heatmap#vValues}) + * @param {number[]} [_=none] + * @returns {heatmap | number[]} + * @memberof heatmap * @property - * by default domainPadding = 0.5 + * by default vValues = undefined */ - violin.domainPadding = function(_) { return arguments.length ? (domainPadding = _, violin) : domainPadding; }; + hm.vValues = function(_) { return arguments.length ? (vValues = _, hm) : vValues; }; /** - * Gets / sets objectSpacer - * (see {@link violin#objectSpacer}) - * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin + * Gets or sets the xExtractor + * (see {@link heatmap#xExtractor}) + * @param {function} [_=none] + * @returns {heatmap | function} + * @memberof heatmap * @property - * by default objectSpacer = 0.05 + * by default xExtractor = undefined */ - violin.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, violin) : objectSpacer; }; + hm.xExtractor = function(_) { return arguments.length ? (xExtractor = _, hm) : xExtractor; }; /** - * Gets / sets the minObjectSize - * (see {@link violin#minObjectSize}) - * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin + * Gets or sets the yExtractor + * (see {@link heatmap#yExtractor}) + * @param {function} [_=none] + * @returns {heatmap | function} + * @memberof heatmap * @property - * by default minObjectSize = 15 + * by default yExtractor = undefined */ - violin.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, violin) : minObjectSize; }; + hm.yExtractor = function(_) { return arguments.length ? (yExtractor = _, hm) : yExtractor; }; /** - * Gets / sets the maxObjectSize - * (see {@link violin#maxObjectSize}) - * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin + * Gets or sets the vExtractor + * (see {@link heatmap#vExtractor}) + * @param {function} [_=none] + * @returns {heatmap | function} + * @memberof heatmap * @property - * by default maxObjectSize = 50 + * by default vExtractor = undefined */ - violin.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, violin) : maxObjectSize; }; + hm.vExtractor = function(_) { return arguments.length ? (vExtractor = _, hm) : vExtractor; }; /** - * Gets / sets the objectStrokeWidth - * (see {@link violin#objectStrokeWidth}) + * Gets / sets whether or not heatmap is allowed to go beyond specified dimensions + * (see {@link heatmap#spaceX}) + * @param {boolean} [_=none] + * @returns {heatmap | boolean} + * @memberof heatmap + * @property + * by default overflowQ = false + */ + hm.overflowQ = function(_) { return arguments.length ? (overflowQ = _, hm) : overflowQ; }; + /** + * Gets / sets objectSpacer + * (see {@link heatmap#objectSpacer}) * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin + * @returns {heatmap | number} + * @memberof heatmap * @property - * by default objectStrokeWidth = 2 + * by default objectSpacer = 0.0 */ - violin.objectStrokeWidth = function(_) { return arguments.length ? (objectStrokeWidth = _, violin) : objectStrokeWidth; }; - - + hm.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, objectSpacer) : data; }; /** - * Gets / sets the colorFunction - * (see {@link violin#colorFunction}) - * @param {colorFunction} [_=none] - * @returns {violin | colorFunction} - * @memberof violin + * Gets / sets the minObjectSize + * (see {@link heatmap#minObjectSize}) + * @param {number} [_=none] + * @returns {heatmap | number} + * @memberof heatmap * @property - * by default colorFunction = colorFunction() + * by default minObjectSize = 50 */ - violin.colorFunction = function(_) { return arguments.length ? (colorFunction$$1 = _, violin) : colorFunction$$1; }; + hm.yMinObjectSize = function(_) { return arguments.length ? (yMinObjectSize = _, hm) : yMinObjectSize; }; /** - * Gets / sets the colorFunction - * (see {@link violin#colorFunction}) - * @param {colorFunction} [_=none] - * @returns {violin | colorFunction} - * @memberof violin + * Gets / sets the maxObjectSize + * (see {@link heatmap#maxObjectSize}) + * @param {number} [_=none] + * @returns {heatmap | number} + * @memberof heatmap * @property - * by default colorFunction = colorFunction() + * by default maxObjectSize = 100 */ - violin.pointColorFunc = function(_) { return arguments.length ? (pointColorFunc = _, violin) : pointColorFunc; }; - - + hm.yMaxObjectSize = function(_) { return arguments.length ? (yMaxObjectSize = _, hm) : yMaxObjectSize; }; /** - * Gets / sets the pointRadius - * (see {@link violin#pointRadius}) + * Gets / sets the minObjectSize + * (see {@link heatmap#minObjectSize}) * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin + * @returns {heatmap | number} + * @memberof heatmap * @property - * by default pointRadius = 2 + * by default minObjectSize = 50 */ - violin.pointRadius = function(_) { return arguments.length ? (pointRadius = _, violin) : pointRadius; }; + hm.xMinObjectSize = function(_) { return arguments.length ? (xMinObjectSize = _, hm) : xMinObjectSize; }; /** - * Gets / sets the pointStrokeWidth - * (see {@link violin#pointStrokeWidth}) + * Gets / sets the maxObjectSize + * (see {@link heatmap#maxObjectSize}) * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin + * @returns {heatmap | number} + * @memberof heatmap * @property - * by default pointStrokeWidth = 2 + * by default maxObjectSize = 100 */ - violin.pointStrokeWidth = function(_) { return arguments.length ? (pointStrokeWidth = _, violin) : pointStrokeWidth; }; - - + hm.xMaxObjectSize = function(_) { return arguments.length ? (xMaxObjectSize = _, hm) : xMaxObjectSize; }; + /** + * Gets / sets the objectStrokeWidth + * (see {@link heatmap#objectStrokeWidth}) + * @param {number} [_=none] + * @returns {heatmap | number} + * @memberof heatmap + * @property + * by default objectStrokeWidth = 2 + */ + hm.objectStrokeWidth = function(_) { return arguments.length ? (objectStrokeWidth = _, hm) : objectStrokeWidth; }; /** * Gets / sets the backgroundFill - * (see {@link violin#backgroundFill}) + * (see {@link heatmap#backgroundFill}) * @param {string} [_=none] - * @returns {violin | string} - * @memberof violin + * @returns {heatmap | string} + * @memberof heatmap * @property * by default backgroundFill = 'transparent' */ - violin.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, violin) : backgroundFill; }; + hm.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, hm) : backgroundFill; }; /** * Gets / sets the namespace - * (see {@link violin#namespace}) + * (see {@link heatmap#namespace}) * @param {string} [_=none] - * @returns {violin | string} - * @memberof violin + * @returns {heatmap | string} + * @memberof heatmap * @property - * by default namespace = 'd3sm-violin' + * by default namespace = 'd3sm-heatmap' */ - violin.namespace = function(_) { return arguments.length ? (namespace = _, violin) : namespace; }; + hm.namespace = function(_) { return arguments.length ? (namespace = _, hm) : namespace; }; /** * Gets / sets the objectClass - * (see {@link violin#objectClass}) + * (see {@link heatmap#objectClass}) * @param {string} [_=none] - * @returns {violin | string} - * @memberof violin + * @returns {heatmap | string} + * @memberof heatmap * @property * by default objectClass = 'tick-group' */ - violin.objectClass = function(_) { return arguments.length ? (objectClass = _, violin) : objectClass; }; - - + hm.objectClass = function(_) { return arguments.length ? (objectClass = _, hm) : objectClass; }; /** * Gets / sets the transitionDuration - * (see {@link violin#transitionDuration}) + * (see {@link heatmap#transitionDuration}) * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin + * @returns {heatmap | number} + * @memberof heatmap * @property * by default transitionDuration = 1000 */ - violin.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, violin) : transitionDuration; }; + hm.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, hm) : transitionDuration; }; /** * Gets / sets the easeFunc - * (see {@link violin#easeFunc}) + * (see {@link heatmap#easeFunc}) * @param {d3.ease} [_=none] - * @returns {violin | d3.ease} - * @memberof violin + * @returns {heatmap | d3.ease} + * @memberof heatmap * @property * by default easeFunc = d3.easeExp */ - violin.easeFunc = function(_) { return arguments.length ? (easeFunc = _, violin) : easeFunc; }; - + hm.easeFunc = function(_) { return arguments.length ? (easeFunc = _, hm) : easeFunc; }; /** - * Gets / sets the quartileKey - * (see {@link violin#quartileKey}) - * @param {string} [_=none] - * @returns {violin | string} - * @memberof violin + * Gets / sets the tooltip + * (see {@link heatmap#tooltip}) + * @param {tooltip} [_=none] + * @returns {heatmap | tooltip} + * @memberof heatmap * @property - * by default quartileKey = "utils.math.quartiles" + * by default tooltip = tooltip() */ - violin.quartileKey = function(_) { return arguments.length ? (quartileKey = _, violin) : quartileKey; }; + hm.tooltip = function(_) { return arguments.length ? (tooltip$$1 = _, hm) : tooltip$$1; }; + /** - * Gets / sets the quartileKeys - * (see {@link violin#quartileKeys}) - * @param {string[]} [_=none] - * @returns {violin | string[]} - * @memberof violin + * Gets / sets the colorFunction + * (see {@link heatmap#colorFunction}) + * @param {colorFunction} [_=none] + * @returns {heatmap | colorFunction} + * @memberof heatmap * @property - * by default quartileKeys = ["Q0","Q1","Q2","Q3","Q4"] + * by default colorFunction = colorFunction() */ - violin.quartileKeys = function(_) { return arguments.length ? (quartileKeys = _, violin) : quartileKeys; }; - + hm.colorFunction = function(_) { return arguments.length ? (colorFunction$$1 = _, hm) : colorFunction$$1; }; /** - * Gets / sets the violinKeys - * (see {@link violin#violinKeys}) - * @param {string[]} [_=none] - * @returns {violin | string[]} - * @memberof violin + * Gets / sets the xSize + * (see {@link heatmap#xSize}) + * @param {number} [_=none] + * @returns {heatmap | number} + * @memberof heatmap * @property - * by default violinKeys = undefined + * by default xSize = undefined */ - violin.violinKeys = function(_) { return arguments.length ? (violinKeys = _, violin) : violinKeys; }; + hm.xSize = function(_) { return arguments.length ? (xSize = _, hm) : xSize; }; /** - * Gets / sets the violinValues - * (see {@link violin#violinValues}) - * @param {Object[]} [_=none] - * @returns {violin | Object[]} - * @memberof violin + * Gets / sets the xSpacerSize + * (see {@link heatmap#xSpacerSize}) + * @param {number} [_=none] + * @returns {heatmap | number} + * @memberof heatmap * @property - * by default violinValues = undefined + * by default xSpacerSize = undefined */ - violin.violinValues = function(_) { return arguments.length ? (violinValues = _, violin) : violinValues; }; - + hm.xSpacerSize = function(_) { return arguments.length ? (xSpacerSize = _, hm) : xSpacerSize; }; /** - * Gets / sets the objectSize - * (see {@link violin#objectSize}) + * Gets / sets the ySize + * (see {@link heatmap#ySize}) * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin + * @returns {heatmap | number} + * @memberof heatmap * @property - * by default objectSize = undefined + * by default ySize = undefined */ - violin.objectSize = function(_) { return arguments.length ? (objectSize = _, violin) : objectSize; }; + hm.ySize = function(_) { return arguments.length ? (ySize = _, hm) : ySize; }; /** - * Gets / sets the spacerSize - * (see {@link violin#spacerSize}) + * Gets / sets the ySpacerSize + * (see {@link heatmap#ySpacerSize}) * @param {number} [_=none] - * @returns {violin | number} - * @memberof violin + * @returns {heatmap | number} + * @memberof heatmap * @property - * by default spacerSize = undefined + * by default ySpacerSize = undefined */ - violin.spacerSize = function(_) { return arguments.length ? (spacerSize = _, violin) : spacerSize; }; - /** - * Gets / sets the tooltip - * (see {@link violin#tooltip}) - * @param {tooltip} [_=none] - * @returns {violin | tooltip} - * @memberof violin - * @property - * by default tooltip = tooltip() - */ - violin.tooltip = function(_) { return arguments.length ? (tooltip$$1 = _, violin) : tooltip$$1; }; - /** - * Gets / sets the pointsTooltip - * (see {@link violin#pointsTooltip}) - * @param {tooltip} [_=none] - * @returns {violin | tooltip} - * @memberof violin - * @property - * by default pointsTooltip = tooltip() - */ - violin.pointsTooltip = function(_) { return arguments.length ? (pointsTooltip = _, violin) : pointsTooltip; }; - - - + hm.ySpacerSize = function(_) { return arguments.length ? (ySpacerSize = _, hm) : ySpacerSize; }; + // hm.yKeySortingFunction = function(_) { return arguments.length ? (yKeySortingFunction = _, hm) : yKeySortingFunction; } + // hm.xKeySortingFunction = function(_) { return arguments.length ? (xKeySortingFunction = _, hm) : xKeySortingFunction; } - function violin () { - // for convenience in handling orientation specific values - var horizontalQ = (orient == 'horizontal') ? true : false; - // background cliping rectangle + function hm() { var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); - // if grouping is undefined sort violinKeys by sortingFunction - var ordered = (grouping == undefined) ? d3.keys(data).sort(sortingFunction) : grouping; - - // console.log(ordered) - - violinKeys = utils.arr.flatten(ordered); - + cellKeys = d3.keys(data); - var calcValues = neededViolinValues() - .horizontalQ(horizontalQ) - .quartileKeys(quartileKeys) - .violinPointsExtractor(violinPointsExtractor) - .violinPointValueExtractor(violinPointValueExtractor); + xValues = utils.arr.unique(cellKeys.map(xExtractor)); + yValues = utils.arr.unique(cellKeys.map(yExtractor)); + vValues = utils.arr.unique(cellKeys.map(vExtractor)); + cellKeys.sort(function(a, b){ return xKeySortingFunction(a, b) || yKeySortingFunction(a, b) }); + utils.con.log('heatmap', 'cells are sorted by', cellKeys); - // augment valus - violinKeys.map(function(vk, i){ calcValues(vk, data); }); - var numberOfObjects = violinKeys.length; + utils.con.log('heatmap', 'x and y keys are', {x: xValues, y:yValues}); - var min = [].concat(...violinKeys.map(function(k, i){return data[k].utils.math.quartiles[quartileKeys[0]]})); - var max = [].concat(...violinKeys.map(function(k, i){return data[k].utils.math.quartiles[quartileKeys[quartileKeys.length - 1]]})); - var extent = [Math.min(...min) - domainPadding, Math.max(...max) + domainPadding]; - // console.log(extent, violinValues, ordered) - // set the scale - scale.domain(extent).range(horizontalQ ? [0,spaceY] : [0, spaceX]); - var space = horizontalQ ? spaceX : spaceY; - // calculate object size - objectSize = utils.math.calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ); - // calculate spacer size if needed - spacerSize = utils.math.calculateWidthOfSpacer(ordered, space, objectSize, numberOfObjects, objectSpacer, overflowQ); - // make the nested groups - var spacerFunction = groupingSpacer() - .horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects) - .objectClass(objectClass).objectSize(objectSize).spacerSize(spacerSize) - .transitionDuration(transitionDuration).easeFunc(easeFunc) - .namespace(namespace); + var xDim = xValues.length, yDim = yValues.length; - // move stuff - spacerFunction(container, ordered, 0); - // console.log(violinKeys, ordered, container.selectAll('g:not(.to-remove).'+objectClass).nodes()) - // for color function - var parentIndexArray = []; - container.selectAll('g:not(.to-remove).'+objectClass) - .each(function(d, i){if (utils.arr.hasQ(violinKeys, d)){ parentIndexArray.push(Number(d3.select(this).attr('parent-index')));}}); + ySize = utils.math.calculateWidthOfObject(spaceY, yDim, yMinObjectSize, yMaxObjectSize, objectSpacer, overflowQ); + xSize = utils.math.calculateWidthOfObject(spaceX, xDim, xMinObjectSize, xMaxObjectSize, objectSpacer, overflowQ); + ySpacerSize = utils.math.calculateWidthOfSpacer(yValues, spaceY, ySize, yDim, objectSpacer, overflowQ); + xSpacerSize = utils.math.calculateWidthOfSpacer(xValues, spaceX, xSize, xDim, objectSpacer, overflowQ); + // console.table({ + // x:{ + // object: xSize, + // spacer: xSpacerSize, + // dim: xDim + // }, + // y:{ + // object: ySize, + // spacer: ySpacerSize, + // dim: yDim + // } + // + // }) + utils.con.log('heatmap', 'size of', {x: xSize, y: ySize}); - // update color function - colorFunction$$1 = colorFunction$$1.colorBy() == 'index' - ? colorFunction$$1.dataExtent([0, Math.max(...parentIndexArray)]) - : colorFunction$$1.dataExtent(extent); - /* violiin specific needs */ + var ySpacer = groupingSpacer() + .horizontalQ(false) + .moveby('category') + .numberOfObjects(yDim) + .objectClass(utils.str.hypenate(objectClass, 'row')) + .objectSize(ySize + ySpacerSize) + .spacerSize(0) + .transitionDuration(transitionDuration) + .easeFunc(easeFunc) + .namespace('row'); - var frequencyMax = Math.max(...[].concat(...violinKeys.map(function(k, i){return d3.max(data[k].frequencies)}))); - var vScale = d3.scaleLinear().domain([0, frequencyMax]).range([0, objectSize / 2]); + var xSpacer = groupingSpacer() + .horizontalQ(true) + .moveby('category') + .numberOfObjects(xDim) + .objectClass(objectClass) + .objectSize(xSize + xSpacerSize) + .spacerSize(0) + .transitionDuration(transitionDuration) + .easeFunc(easeFunc); - var lArea = d3.line() - .x(function(d, i){ return horizontalQ ? -vScale(d.x) : scale(d.x)}) - .y(function(d, i){ return horizontalQ ? scale(extent[1]) - scale(d.y) : -vScale(d.y)}) - .curve(d3.curveBasis); - var rArea = d3.line() - .x(function(d, i){ return horizontalQ ? vScale(d.x) : scale(d.x)}) - .y(function(d, i){ return horizontalQ ? scale(extent[1]) - scale(d.y) : vScale(d.y)}) - .curve(d3.curveBasis); + ySpacer(container, yValues, 0); + container.selectAll('g.'+utils.str.hypenate(objectClass, 'row')) + .each(function(d, i){ xSpacer(d3.select(this), xValues, 0); }); + var cells = container.selectAll('g:not(.to-remove).'+objectClass); + if (cellKeys.length != yValues.length * xValues.length) { + var lookup = {}; + cellKeys.map(function(k, i){ + lookup[xExtractor(k)+'::'+yExtractor(k)] = k; + }); + var positionedCellKeys = []; + for (var i = 0; i < yValues.length; i++) { + for (var j = 0; j < xValues.length; j++) { + var lookupValue = lookup[xValues[j]+"::"+yValues[i]]; + if (lookupValue == undefined) { + positionedCellKeys.push(undefined); + } else { + positionedCellKeys.push(lookupValue); + } + } + } - container.selectAll('g:not(.to-remove).'+objectClass).each(function(key, i){ - var t = d3.select(this), - currentData = data[key]; - // needed because bug in selecting .to-remove - if (!utils.arr.hasQ(violinKeys, key)) {return} - var - i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'), - fillColor = colorFunction$$1(key, currentData, i, 'fill'), // prevent duplicate computation - strokeColor = colorFunction$$1(key, currentData, i, 'stroke'), - area = utils.sel.safeSelect(t, 'g', 'area'), - la = utils.sel.safeSelect(area, 'path', 'left'), - ra = utils.sel.safeSelect(area, 'path', 'right'), - quarts = utils.sel.safeSelect(t, 'g', 'quarts'), - lq3 = utils.sel.safeSelect(quarts, 'line', 'q3'), - lq1 = utils.sel.safeSelect(quarts, 'line', 'q1'), - q3 = currentData.utils.math.quartiles[quartileKeys[3]], - q2 = currentData.utils.math.quartiles[quartileKeys[2]], - q1 = currentData.utils.math.quartiles[quartileKeys[1]]; + cells.data(positionedCellKeys); - t.attr('transform', horizontalQ ? 'translate('+objectSize / 2+',0)' : 'translate(0,'+objectSize / 2+')' ); - // draw curve - la.transition().duration(transitionDuration).attr('d', function(dd, ii){ return lArea(currentData.contour)}) - .attr('fill', fillColor) - .attr('stroke', strokeColor) - .attr('stroke-width', objectStrokeWidth); + // maybe breaks this + // !!!!!! IMPORTANT NOTE TODO LOOK HERE BUG + cellKeys = positionedCellKeys; + } else { + cells.data(cellKeys); + } - ra.transition().duration(transitionDuration).attr('d', function(dd, ii){ return rArea(currentData.contour)}) - .attr('fill', fillColor) - .attr('stroke', strokeColor) - .attr('stroke-width', objectStrokeWidth); - area.node().addEventListener('mouseover', function(dd, ii){ - container.selectAll('g.'+objectClass).style('opacity', 0.2); - t.style('opacity', 1); - la.attr('stroke-width',objectStrokeWidth*2); - ra.attr('stroke-width',objectStrokeWidth*2); - }); - area.node().addEventListener('mouseout', function(dd, ii){ - container.selectAll('g.'+objectClass).style('opacity', 1); - la.attr('stroke-width',objectStrokeWidth); - ra.attr('stroke-width',objectStrokeWidth); - }); - if (pointsQ) { - var ptsContainer = utils.sel.safeSelect(t, 'g', 'points'); - var pts = ptsContainer.selectAll('.point').data(currentData.pointKeys); - pts.on('mouseover', null); + var parentIndexArray = []; + cells.each(function(d, i){ parentIndexArray.push(Number(d3.select(this).attr('parent-index'))); }); + colorFunction$$1 = colorFunction$$1.colorBy() == 'index' + ? colorFunction$$1.dataExtent([0, Math.max(...parentIndexArray)]) + : colorFunction$$1.dataExtent([0, Math.max(...vValues)]); - var ptsExit = pts.exit().transition().ease(easeFunc).duration(transitionDuration) - .attr('r', 0) - .attr('cy', horizontalQ ? scale(extent[1]) - scale(q2) : vScale(0)) - .attr('cx', horizontalQ ? vScale(0) : scale(q2)).remove(); + cells.each(function(key, i) { + utils.con.log('heatmap', 'each cell', {key: key, index: i, node: d3.select(this).node()}); - var ptsEnter = pts.enter().append('circle').attr('class', 'point').attr('r', 0) - .attr('cx', horizontalQ ? 0 : scale(q2)) - .attr('cy', horizontalQ ? scale(q2) : 0); + var t = d3.select(this); + if (key == undefined) {return} + var currentData = data[key], + value = vExtractor(key, i), + i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'), + fillColor = colorFunction$$1(key, value, i, 'fill'), // prevent duplicate computation + strokeColor = colorFunction$$1(key, value, i, 'stroke'); - pts = pts.merge(ptsEnter); + var c = utils.sel.safeSelect(t, 'rect', utils.str.hypenate(objectClass,'rect')); + c.attr('width', xSize + xSpacerSize - objectStrokeWidth) + .attr('height', ySize + ySpacerSize - objectStrokeWidth) + .attr('fill', fillColor) + .attr('x', objectStrokeWidth/2) + .attr('y', objectStrokeWidth/2) + .attr('stroke', "#000") + .attr('stroke-width', objectStrokeWidth); - // console.log(pointsTooltip.header()) + }); - var pTTips = tooltip().selection(pts).data(violinPointsExtractor(key, currentData)) - .header(pointsTooltip.header()) - .keys(pointsTooltip.keys()) - .values(pointsTooltip.values()); + tooltip$$1.selection(cells.selectAll('rect.'+utils.str.hypenate(objectClass, 'rect'))) + .data(data); + // .keys(['r', 'v']) + // .header(function(d, i){return utils.str.hypenate(data[d][xKey], data[d][yKey]) }) - pTTips(); - - var pMin = d3.min(currentData.pointValues), pMax = d3.max(currentData.pointValues); - - - - pts.transition().duration(transitionDuration).ease(easeFunc).attr('r', pointRadius) - .attr('cy', function(pointKey, ii){ - var dd = currentData.pointValues[ii]; - if (horizontalQ) { return scale(extent[1]) - scale(dd) } - var j = utils.arr.whichBin(currentData.binned, dd); - var r = Math.random(); - var n = vScale(r * currentData.frequencies[j] * 0.5); - var k = Math.random() > 0.5 ? n : -n; - return k - }) - .attr('cx', function(pointKey, ii){ - var dd = currentData.pointValues[ii]; - if (horizontalQ) { - var j = utils.arr.whichBin(currentData.binned, dd); - var r = Math.random(); - var n = vScale(r * currentData.frequencies[j] * 0.5); - var k = Math.random() > 0.5 ? n : -n; - return k - } - return scale(dd) - }) - .attr('stroke', function(dd, ii) { var dd = currentData.pointValues[ii]; return pointColorFunc(dd, 'stroke', strokeColor, pMin, pMax) }) - .attr('fill' , function(dd, ii) { var dd = currentData.pointValues[ii]; return pointColorFunc(dd, 'fill' , strokeColor, pMin, pMax) }) - .attr('stroke-width', pointStrokeWidth); - - ptsContainer.selectAll('circle.point').on('mouseover', function(dd, ii){ - container.selectAll('g.'+objectClass).style('opacity', 0.2); - t.style('opacity', 1); - la.attr('stroke-width',objectStrokeWidth*2); - ra.attr('stroke-width',objectStrokeWidth*2); - - container.selectAll('.point').style('opacity', 0.2); - d3.select(this).style('opacity', 1).attr('r', pointRadius * 2).attr('stroke-width',pointStrokeWidth*2); - }); - ptsContainer.selectAll('circle.point').on('mouseout', function(dd, ii){ - var e = document.createEvent('SVGEvents'); - e.initEvent('mouseout',true,true); - area.node().dispatchEvent(e); - - container.selectAll('.point').style('opacity', 1); - d3.select(this).attr('stroke-width', pointStrokeWidth).attr('r', pointRadius); - }); - } - else { - cV.selectAll('.point') - .transition().duration(transitionDuration).ease(easeFunc) - .attr('r', 0) - .attr('cy', horizontalQ ? scale(extent[1]) - scale(q2) : vScale(0)) - .attr('cx', horizontalQ ? vScale(0) : scale(q2)) - .remove(); - } - - - }); - - - tooltip$$1.selection(container.selectAll('g:not(.to-remove).'+objectClass + ' .area')); - if (tooltip$$1.data() == undefined) {tooltip$$1.data(data);} - tooltip$$1(); - if (tooltip$$1.values() == undefined) { - tooltip$$1.values([ - function(currentData, tooltipKey){ return currentData['utils.math.quartiles'][tooltipKey] }, - function(currentData, tooltipKey){ return currentData['utils.math.quartiles'][tooltipKey] }, - function(currentData, tooltipKey){ return currentData['utils.math.quartiles'][tooltipKey] }, - function(currentData, tooltipKey){ return currentData['utils.math.quartiles'][tooltipKey] }, - function(currentData, tooltipKey){ return currentData['utils.math.quartiles'][tooltipKey] } - ]); - - } - - } - - return violin - } + tooltip$$1(); + } + return hm; + } + /******************************************************************************* + ** ** + ** ** + ** VIOLIN ** + ** ** + ** ** + *******************************************************************************/ /** - * Produces the function which manipulates the violin data to have values needed - * for rendering the violins as svg. - * @returns {function} calculateViolinValues - * @namespace neededViolinValues - */ - function neededViolinValues() { - var + * Creates a violin + * + * {@link https://sumneuron.gitlab.io/d3sm/demos/basic-violins/index.html Demo} + * @constructor violin + * @param {d3.selection} selection + * @namespace violin + * @returns {function} violin + */ + function violin( selection ) { + var /** - * Whether or not the orientation of the violins are horizontal + * Data to plot. Assumed to be a object, where each key corresponds to a violin + * (see {@link violin#data}) + * @param {Object} [data=undefined] + * @memberof violin# + * @property + */ + data, + /** + * Which direction to render the bars in * (see {@link violin#orient}) - * @param {Object} [horizontalQ=true] - * @memberof neededViolinValues# + * @param {number} [orient='horizontal'] + * @memberof violin# * @property */ - horizontalQ = true, + orient='horizontal', /** - * Keys to be put into the utils.math.quartiles if they need to be calculated. - * (see {@link violin#quartileKeys}) - * @param {Object} [quartileKeys=['Q0', 'Q1', 'Q2', 'Q3', 'Q4']] - * @memberof neededViolinValues# + * Amount of horizontal space (in pixels) avaible to render the violin in + * (see {@link violin#spaceX}) + * @param {number} [spaceX=undefined] + * @memberof violin# * @property */ - quartileKeys = ['Q0', 'Q1', 'Q2', 'Q3', 'Q4'], + spaceX, /** - * Function which given the key of the violin and that key's associated value - * returns the object consiting of the points of the violin - * (see {@link violin#violinPointsExtractor}) - * @param {Object} [violinPointsExtractor=function(violinKey, violinData) { return violinData.points }] - * @memberof neededViolinValues# + * Amount of vertical space (in pixels) avaible to render the violin in + * (see {@link violin.spaceY}) + * @param {number} [spaceY=undefined] + * @memberof violin# * @property */ - violinPointsExtractor, + spaceY, /** - * Function which given the key of the current point and the object of points for the - * violin, returns the numerical value of the point - * (see {@link violin#violinPointValueExtractor}) - * @param {Object} [violinPointValueExtractor=function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value }] - * @memberof neededViolinValues# + * Whether or not to allow violin to render elements pass the main spatial dimension + * given the orientation (see {@link violin#orient}), where {@link violin#orient}="horizontal" + * the main dimension is {@link violin#spaceX} and where {@link violin#orient}="vertical" + * the main dimension is {@link violin#spaceY} + * @param {boolean} [overflowQ=false] + * @memberof violin# * @property */ - violinPointValueExtractor; - - + overflowQ = true, /** - * Gets / sets the horizontalQ - * (see {@link violin#orient}) - * @param {boolean} [_=none] - * @returns {calculateViolinValues | boolean} - * @memberof calculateViolinValues - * @property - * - * by default horizontalQ = true - */ - calculateViolinValues.horizontalQ = function(_) { return arguments.length ? (horizontalQ=_, calculateViolinValues) : horizontalQ }; + * Whether or not to display points inside the points + * @param {boolean} [pointsQ=false] + * @memberof violin# + * @property + */ + pointsQ = true, /** - * Gets / sets the quartileKeys - * (see {@link violin#quartileKeys}) - * @param {string[]} [_=none] - * @returns {calculateViolinValues | string[]} - * @memberof calculateViolinValues - * @property - * - * by default quartileKeys = ["Q0","Q1","Q2","Q3","Q4"] - */ - calculateViolinValues.quartileKeys = function(_) { return arguments.length ? (quartileKeys=_, calculateViolinValues) : quartileKeys }; + * An array - putatively of other arrays - depicting how bars should be arranged + * @param {Array[]} [grouping=undefined] + * @memberof violin# + * @property + */ + grouping, /** - * Gets / sets the violinPointsExtractor - * (see {@link violin#violinPointsExtractor}) - * @param {function} [_=none] - * @returns {calculateViolinValues | function} - * @memberof calculateViolinValues - * @property - * - * by default violinPointsExtractor = function(violinKey, violinData) { return violinData.points } - */ - calculateViolinValues.violinPointsExtractor = function(_) { return arguments.length ? (violinPointsExtractor=_, calculateViolinValues) : violinPointsExtractor }; + * How to get the value of the violin + * @param {function} [valueExtractor=function(key, index) { return data[key] }] + * @memberof violin# + * @property + */ + valueExtractor = function(key, index) {return data[key] }, /** - * Gets / sets the violinPointValueExtractor - * (see {@link violin#violinPointValueExtractor}) - * @param {function} [_=none] - * @returns {calculateViolinValues | function} - * @memberof calculateViolinValues - * @property - * - * by default violinPointValueExtractor = function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value } - */ - calculateViolinValues.violinPointValueExtractor = function(_) { return arguments.length ? (violinPointValueExtractor=_, calculateViolinValues) : violinPointValueExtractor }; - + * How to sort the bars - if {@link violin#grouping} is not provided. + * @param {function} [sortingFunction=function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}] + * @memberof violin# + * @property + */ + sortingFunction = function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}, /** - * The function produced by neededViolinValues. - * - * Adds the data need to render the violin as an svg - * @param {string} violinKey the key of the current violin - * @param {object} data the object consisting of violinKey - values (violin data) pairs - * @returns {none} this function manipulates the passed data object adding the following keys - * - * data[violinKey].binned // the binned values of the points - * - * data[violinKey].frequencies // the list consisting of the length of each bin - * - * data[violinKey].contour // the points depicting the contour of 1/2 of the violin - * - * data[violinKey].utils.math.quartiles // the utils.math.quartiles of the points' values - * - * data[violinKey].pointKeys // the keys associated with the points - * - * data[violinKey].pointValues // the numerical values of the points - * - * @memberof neededViolinValues# + * The scale for which violin values should be transformed by + * @param {d3.scale} [scale=d3.scaleLinear] + * @memberof violin# * @property */ - function calculateViolinValues(violinKey, data) { - // data for the current violin - var violinData = data[violinKey]; - // the object of points - var violinPoints = violinPointsExtractor(violinKey, violinData); - // - var violinPointsKeys = d3.keys(violinPoints); - // the numerical values of those points - var violinPointsValues = violinPointsKeys.map(function(pk, i){return violinPointValueExtractor(pk, violinPoints)}); - - // utils.math.quartiles of those points - var pointQuartiles = utils.math.quartiles(violinPointsValues, quartileKeys); + scale = d3.scaleLinear(), + /** + * The padding for the domain of the scale (see {@link violin#scale}) + * @param {number} [domainPadding=0.5] + * @memberof violin# + * @property + */ + domainPadding = 0.5, + /** + * Default space for the spacer (percentage) of main dimension given the orientation + * (see {@link violin#orient}), where {@link violin#orient}="horizontal" + * the main dimension is {@link violin#spaceX} and where {@link violin#orient}="vertical" + * the main dimension is {@link violin#spaceY} between bars + * @param {number} [objectSpacer=0.05] + * @memberof violin# + * @property + */ + objectSpacer = 0.05, + /** + * The minimum size that an object can be + * @param {number} [minObjectSize=50] + * @memberof violin# + * @property + */ + minObjectSize = 50, + /** + * The maximum size that an object can be + * @param {number} [maxObjectSize=100] + * @memberof violin# + * @property + */ + maxObjectSize = 100, - // binned points - var binned = d3.histogram()(violinPointsValues); - // length of bins - var frequencies = binned.map(bin=>bin.length); - // min and max countour points for nice drawings - var minContourPoint = horizontalQ ? {x: 0, y: d3.min(violinPointsValues)} : {x: d3.min(violinPointsValues), y: 0}; - var maxContourPoint = horizontalQ ? {x: 0, y: d3.max(violinPointsValues)} : {x: d3.max(violinPointsValues), y: 0}; - var violinContourPoints = binned.map(function(bin, i) { - return horizontalQ - ? {y: (bin.length) ? d3.median(bin): d3.median([bin.x0, bin.x1]), x: frequencies[i]} - : {x: (bin.length) ? d3.median(bin): d3.median([bin.x0, bin.x1]), y: frequencies[i]} - }); - // points along which to draw the violin shpe - violinContourPoints = [minContourPoint].concat(violinContourPoints).concat([maxContourPoint]); - - // set data - violinData.binned = binned; - violinData.frequencies = frequencies; - violinData.contour = violinContourPoints; - violinData.utils.math.quartiles = pointQuartiles; - violinData.pointKeys = violinPointsKeys; - violinData.pointValues = violinPointsValues; - } + /** + * The stroke width of the bars + * @param {number} [barStrokeWidth=2] + * @memberof violin# + * @property + */ + objectStrokeWidth = 2, + /** + * Instance of ColorFunction + * @param {function} [colorFunction = colorFunction()] + * @memberof violin# + * @property + */ + colorFunction$$1 = colorFunction(), + /** + * Instance of ColorFunction modified by a scale for the points + * @param {function} [pointColorFunc = colorFunction()] + * @memberof violin# + * @property + */ + pointColorFunc = function (d, type, base, min, max) { + var minMaxHexScale = d3.scaleLinear().domain([min, max]).range([-0.25, 0.05]); + var scaledColor = utils.color.modifyHexidecimalColorLuminance(base.replace('#', ''), minMaxHexScale(d)); + var mod = type == "stroke" ? 0 : 0.25; + return utils.color.modifyHexidecimalColorLuminance(scaledColor.replace('#', ''), mod) + }, - return calculateViolinValues - } + /** + * The radius of a point + * @param {number} [pointRadius=3] + * @memberof violin# + * @property + */ + pointRadius = 3, + /** + * The stroke width of the oints + * @param {number} [pointStrokeWidth=2] + * @memberof violin# + * @property + */ + pointStrokeWidth = 2, - function upset ( selection ) { - var - data, - orient='horizontal', - spaceX, - spaceY, - overflowQ=false, - minObjectSize=20, - maxObjectSize=50, - circleStrokeWidth=2, - // colorFunction= + /** + * Color of the background + * @param {string} [backgroundFill="transparent"] + * @memberof violin# + * @property + */ backgroundFill = 'transparent', - namespace='d3sm-upset', - objectClass = 'upset', - + /** + * Namespace for all items made by this instance of violin + * @param {string} [namespace="d3sm-violin"] + * @memberof violin# + * @property + */ + namespace = 'd3sm-violin', + /** + * Class name for violin container ( element) + * @param {string} [objectClass="violin"] + * @memberof violin# + * @property + */ + objectClass = 'violin', + /** + * Duration of all transitions of this element + * @param {number} [transitionDuration=1000] + * @memberof violin# + * @property + */ transitionDuration = 1000, + /** + * Easing function for transitions + * @param {d3.ease} [easeFunc=d3.easeExp] + * @memberof violin# + * @property + */ easeFunc = d3.easeExp, - setKey = "set", - intersectionKey = "intersection", - elementsKey = "elements", - - setExtractor = function(key, i) {return data[key][setKey]}, - intersectionExtractor = function(key, i) {return data[key][intersectionKey]}, - elementExtractor = function(key, i) {return data[key][elementsKey]}, - - cellKeys, - setValues, - intersectionValues, - xObjectSpacer = 0.05, - yObjectSpacer = 0.05, - radius, - - // listDelim = ';' - - yObjectSize, - ySpacerSize, - xObjectSize, - xSpacerSize, - - setKeySortingFunction = function(a, b) { return setValues.indexOf(setExtractor(a)) - setValues.indexOf(setExtractor(b)) }, - intersectionKeySortingFunction = function(a, b) { return intersectionValues.indexOf(intersectionExtractor(a)) - intersectionValues.indexOf(intersectionExtractor(b)) }; - - - upset.selection = function(_) { return arguments.length ? (selection = _, upset) : selection; }; - upset.data = function(_) { return arguments.length ? (data = _, upset) : data; }; - upset.orient = function(_) { return arguments.length ? (orient = _, upset) : orient; }; - upset.spaceX = function(_) { return arguments.length ? (spaceX = _, upset) : spaceX; }; - upset.spaceY = function(_) { return arguments.length ? (spaceY = _, upset) : spaceY; }; - upset.overflowQ = function(_) { return arguments.length ? (overflowQ = _, upset) : overflowQ; }; - upset.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, upset) : minObjectSize; }; - upset.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, upset) : maxObjectSize; }; - upset.circleStrokeWidth = function(_) { return arguments.length ? (circleStrokeWidth = _, upset) : circleStrokeWidth; }; - upset.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, upset) : backgroundFill; }; - upset.namespace = function(_) { return arguments.length ? (namespace = _, upset) : namespace; }; - upset.objectClass = function(_) { return arguments.length ? (objectClass = _, upset) : objectClass; }; - upset.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, upset) : transitionDuration; }; - upset.easeFunc = function(_) { return arguments.length ? (easeFunc = _, upset) : easeFunc; }; - upset.cellKeys = function(_) { return arguments.length ? (cellKeys = _, upset) : cellKeys; }; - upset.setValues = function(_) { return arguments.length ? (setValues = _, upset) : setValues; }; - upset.intersectionValues = function(_) { return arguments.length ? (intersectionValues = _, upset) : intersectionValues; }; - upset.xObjectSpacer = function(_) { return arguments.length ? (xObjectSpacer = _, upset) : xObjectSpacer; }; - upset.yObjectSpacer = function(_) { return arguments.length ? (yObjectSpacer = _, upset) : yObjectSpacer; }; - upset.radius = function(_) { return arguments.length ? (radius = _, upset) : radius; }; - upset.setExtractor = function(_) { return arguments.length ? (setExtractor = _, upset) : setExtractor; }; - upset.intersectionExtractor = function(_) { return arguments.length ? (intersectionExtractor = _, upset) : intersectionExtractor; }; - upset.elementExtractor = function(_) { return arguments.length ? (elementExtractor = _, upset) : elementExtractor; }; - upset.setKeySortingFunction = function(_) { return arguments.length ? (setKeySortingFunction = _, upset) : setKeySortingFunction; }; - upset.intersectionKeySortingFunction = function(_) { return arguments.length ? (intersectionKeySortingFunction = _, upset) : intersectionKeySortingFunction; }; - - upset.yObjectSize = function(_) { return arguments.length ? (yObjectSize = _, upset) : yObjectSize; }; - upset.ySpacerSize = function(_) { return arguments.length ? (ySpacerSize = _, upset) : ySpacerSize; }; - upset.xObjectSize = function(_) { return arguments.length ? (xObjectSize = _, upset) : xObjectSize; }; - upset.xSpacerSize = function(_) { return arguments.length ? (xSpacerSize = _, upset) : xSpacerSize; }; - - function upset() { - // for convenience in handling orientation specific values - var horizontalQ = (orient == 'horizontal') ? true : false; - var verticalQ = !horizontalQ; + /** + * The keys corresponding to each quartile + * @param {string[]} [quartileKeys=["Q0", "Q1", "Q2", "Q3", "Q4"]] + * @memberof violin# + * @property + */ + quartileKeys = ["Q0", "Q1", "Q2", "Q3", "Q4"], - // background cliping rectangle - var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; - var container = utils$1.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); + /** + * The keys of the bars + * @param {string[]} [violinKeys=undefined] + * @memberof violin# + * @property + */ + violinKeys, + /** + * The values of the bars + * @param {number[]} [violinValues=undefined] + * @memberof violin# + * @property + */ + violinValues, + /** + * The objectSize (actual width) used by the bars + * @param {number} [objectSize=undefined] + * @memberof violin# + * @property + */ + objectSize, + /** + * The spacerSize (actual width) used by the spacers between the bars + * @param {number} [spacerSize=undefined] + * @memberof violin# + * @property + */ + spacerSize, + /** + * Instance of Tooltip + * @param {function} [tooltip=tooltip()] + * @memberof violin# + * @property + */ + tooltip$$1 = tooltip().keys([quartileKeys[4], quartileKeys[3], quartileKeys[2], quartileKeys[1], quartileKeys[0]]), + pointsTooltip = tooltip(), + // pointKeyExtractor = function(violinKey, violinData, violinValues) {return d3.keys(violinValues[violinKey].values)}, + // pointValueExtractor = function(pointKey, violinKey, violinData, violinValues) {return violinValues[violinKey].values[pointKey]}, - cellKeys = d3.keys(data); - setValues = utils$1.arr.unique(cellKeys.map(setExtractor)).sort(); - intersectionValues = utils$1.arr.unique(cellKeys.map(intersectionExtractor)).sort().sort(function(a, b){ - return a.split(';').length - b.split(';').length - }); - if (!horizontalQ) { - cellKeys.sort(function(a, b){ return setKeySortingFunction(a, b) || intersectionKeySortingFunction(a, b) }); - } else { - cellKeys.sort(function(a, b){ return intersectionKeySortingFunction(a, b) || setKeySortingFunction(a, b) }); - } + /** + * Function which given the key of the violin and that key's associated value + * returns the object consiting of the points of the violin + * (see {@link violin#violinPointsExtractor}) + * @param {Object} [violinPointsExtractor=function(violinKey, violinData) { return violinData.points }] + * @memberof violin# + * @property + */ + violinPointsExtractor = function (violinKey, violinData) {return violinData.points }, + /** + * Function which given the key of the current point and the object of points for the + * violin, returns the numerical value of the point + * (see {@link violin#violinPointValueExtractor}) + * @param {Object} [violinPointValueExtractor=function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value }] + * @memberof violin# + * @property + */ + violinPointValueExtractor = function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value }; + /** + * Gets or sets the violinPointsExtractor + * @param {function} [_=none] + * @returns {violin | function} + * @memberof violin + * @property + * by default violinPointsExtractor = function(violinKey, violinData) { return violinData.points } + */ + violin.violinPointsExtractor = function(_) { return arguments.length ? (violinPointsExtractor = _, violin) : violinPointsExtractor; }; + /** + * Gets or sets the violinPointValueExtractor + * @param {function} [_=none] + * @returns {violin | function} + * @memberof violin + * @property + * by default violinPointsExtractor = function(pointKey, violinKey, violinData, violinValues) {return violinValues[violinKey].values[pointKey]} + */ + violin.violinPointValueExtractor = function(_) { return arguments.length ? (violinPointValueExtractor = _, violin) : violinPointValueExtractor; }; - var - xValues = horizontalQ ? intersectionValues : setValues, - yValues = horizontalQ ? setValues : intersectionValues, - xDim = horizontalQ ? xValues.length : yValues.length, - yDim = horizontalQ ? yValues.length : xValues.length; - - // console.utils.con.log(xValues, yValues) - - - xObjectSize = utils$1.math.calculateWidthOfObject(spaceX, xDim, minObjectSize, maxObjectSize, xObjectSpacer, overflowQ); - yObjectSize = utils$1.math.calculateWidthOfObject(spaceY, yDim, minObjectSize, maxObjectSize, yObjectSpacer, overflowQ); - xSpacerSize = utils$1.math.calculateWidthOfSpacer(xValues, spaceX, xObjectSize, xDim, xObjectSpacer, overflowQ); - ySpacerSize = utils$1.math.calculateWidthOfSpacer(yValues, spaceY, yObjectSize, yDim, yObjectSpacer, overflowQ); - - var ySpacer = groupingSpacer() - .horizontalQ(false) - .moveby('category').numberOfObjects(yDim) - .objectSize(yObjectSize).spacerSize(ySpacerSize) - .transitionDuration(transitionDuration).easeFunc(easeFunc); + /** + * Gets or sets the selection in which items are manipulated + * @param {d3.selection} [_=none] + * @returns {violin | d3.selection} + * @memberof violin + * @property + * by default selection = selection + */ + violin.selection = function(_) { return arguments.length ? (selection = _, violin) : selection; }; + /** + * Gets or sets the data + * (see {@link violin#data}) + * @param {number} [_=none] + * @returns {violin | object} + * @memberof violin + * @property + */ + violin.data = function(_) { return arguments.length ? (data = _, violin) : data; }; + /** + * Gets or sets the orient of the boxes + * (see {@link violin#orient}) + * @param {number} [_=none] + * @returns {violin | object} + * @memberof violin + * @property + */ + violin.orient = function(_) { return arguments.length ? (orient = _, violin) : orient; }; + /** + * Gets or sets the amount of horizontal space in which items are manipulated + * (see {@link violin#spaceX}) + * @param {number} [_=none] should be a number > 0 + * @returns {violin | number} + * @memberof violin + * @property + * by default spaceX = undefined + */ + violin.spaceX = function(_) { return arguments.length ? (spaceX = _, violin) : spaceX; }; + /** + * Gets or sets the amount of vertical space in which items are manipulated + * (see {@link violin#spaceY}) + * @param {number} [_=none] should be a number > 0 + * @returns {violin | number} + * @memberof violin + * @property + * by default spaceY = undefined + */ + violin.spaceY = function(_) { return arguments.length ? (spaceY = _, violin) : spaceY; }; - var xSpacer = groupingSpacer() - .horizontalQ(true) - .moveby('category').numberOfObjects(xDim) - .objectClass(objectClass) - .objectSize(xObjectSize).spacerSize(xSpacerSize) - .transitionDuration(transitionDuration).easeFunc(easeFunc); + /** + * Gets / sets whether or not violin is allowed to go beyond specified dimensions + * (see {@link violin#overflowQ}) + * @param {boolean} [_=none] + * @returns {violin | boolean} + * @memberof violin + * @property + * by default overflowQ = false + */ + violin.overflowQ = function(_) { return arguments.length ? (overflowQ = _, violin) : overflowQ; }; + /** + * Gets / sets whether or not to plot points with the violins + * (see {@link violin#pointsQ}) + * @param {boolean} [_=none] + * @returns {violin | boolean} + * @memberof violin + * @property + * by default pointsQ = false + */ + violin.pointsQ = function(_) { return arguments.length ? (pointsQ = _, violin) : pointsQ; }; - if (verticalQ) { - xSpacer.objectClass(objectClass); - ySpacer.namespace('across').objectClass(utils$1.str.hypenate(objectClass, 'across')); + /** + * Gets / sets the grouping of the boxes + * (see {@link violin#grouping}) + * @param {Array[]} [_=none] + * @returns {violin | Array[]} + * @memberof violin + * @property + * by default grouping = undefined + */ + violin.grouping = function(_) { return arguments.length ? (grouping = _, violin) : grouping; }; + /** + * Gets / sets the valueExtractor + * (see {@link violin#valueExtractor}) + * @param {function} [_=none] + * @returns {violin | function} + * @memberof violin + * @property + */ + violin.valueExtractor = function(_) { return arguments.length ? (valueExtractor = _, violin) : valueExtractor; }; + /** + * Gets / sets the sortingFunction + * (see {@link violin#sortingFunction}) + * @param {function} [_=none] + * @returns {violin | function} + * @memberof violin + * @property + */ + violin.sortingFunction = function(_) { return arguments.length ? (sortingFunction = _, violin) : sortingFunction; }; - ySpacer(container, yValues, 0); - container.selectAll('g.'+utils$1.str.hypenate(objectClass, 'across')) - .each(function(d, i){ xSpacer(d3.select(this), xValues, 0); }); - } else { - xSpacer.namespace('across').objectClass(utils$1.str.hypenate(objectClass, 'across')); - ySpacer.objectClass(objectClass); + /** + * Gets / sets the scale for which the violin values should be transformed by + * (see {@link violin#scale}) + * @param {d3.scale} [_=none] + * @returns {violin | d3.scale} + * @memberof violin + * @property + * by default scale = d3.scaleLinear() + */ + violin.scale = function(_) { return arguments.length ? (scale = _, violin) : scale; }; + /** + * Gets / sets the padding for the domain of the scale + * (see {@link violin#domainPadding}) + * @param {number} [_=none] + * @returns {violin | number} + * @memberof violin + * @property + * by default domainPadding = 0.5 + */ + violin.domainPadding = function(_) { return arguments.length ? (domainPadding = _, violin) : domainPadding; }; - xSpacer(container, xValues, 0); - container.selectAll('g.'+utils$1.str.hypenate(objectClass, 'across')) - .each(function(d, i){ ySpacer(d3.select(this), yValues, 0); }); - } + /** + * Gets / sets objectSpacer + * (see {@link violin#objectSpacer}) + * @param {number} [_=none] + * @returns {violin | number} + * @memberof violin + * @property + * by default objectSpacer = 0.05 + */ + violin.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, violin) : objectSpacer; }; + /** + * Gets / sets the minObjectSize + * (see {@link violin#minObjectSize}) + * @param {number} [_=none] + * @returns {violin | number} + * @memberof violin + * @property + * by default minObjectSize = 15 + */ + violin.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, violin) : minObjectSize; }; + /** + * Gets / sets the maxObjectSize + * (see {@link violin#maxObjectSize}) + * @param {number} [_=none] + * @returns {violin | number} + * @memberof violin + * @property + * by default maxObjectSize = 50 + */ + violin.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, violin) : maxObjectSize; }; - var cells = container.selectAll('g:not(.to-remove).'+objectClass); - var lookup = {}; - cellKeys.map(function(k, i){ - lookup[setExtractor(k)+'::'+intersectionExtractor(k)] = k; - }); + /** + * Gets / sets the objectStrokeWidth + * (see {@link violin#objectStrokeWidth}) + * @param {number} [_=none] + * @returns {violin | number} + * @memberof violin + * @property + * by default objectStrokeWidth = 2 + */ + violin.objectStrokeWidth = function(_) { return arguments.length ? (objectStrokeWidth = _, violin) : objectStrokeWidth; }; - // var positionedCellKeys = [] - // for (var i = 0; i < setValues.length; i++) { - // for (var j = 0; j < intersectionValues.length; j++) { - // var lookupValue = lookup[setValues[j]+"::"+intersectionValues[i]] - // if (lookupValue == undefined) { - // positionedCellKeys.push(undefined) - // } else { - // positionedCellKeys.push(lookupValue) - // } - // console.utils.con.log(i, j, lookupValue) - // } - // } - // - // // console.utils.con.log(positionedCellKeys) - // - // cells.data(positionedCellKeys); + /** + * Gets / sets the colorFunction + * (see {@link violin#colorFunction}) + * @param {colorFunction} [_=none] + * @returns {violin | colorFunction} + * @memberof violin + * @property + * by default colorFunction = colorFunction() + */ + violin.colorFunction = function(_) { return arguments.length ? (colorFunction$$1 = _, violin) : colorFunction$$1; }; + /** + * Gets / sets the colorFunction + * (see {@link violin#colorFunction}) + * @param {colorFunction} [_=none] + * @returns {violin | colorFunction} + * @memberof violin + * @property + * by default colorFunction = colorFunction() + */ + violin.pointColorFunc = function(_) { return arguments.length ? (pointColorFunc = _, violin) : pointColorFunc; }; - cells.data(cellKeys); - cells.each(function(key, i) { - var t = d3.select(this); - if (key == undefined) {return } - var - currentData = data[key]; - // console.utils.con.log(key, currentData) - var - set = setExtractor(key, i), - intersection = intersectionExtractor(key, i); + /** + * Gets / sets the pointRadius + * (see {@link violin#pointRadius}) + * @param {number} [_=none] + * @returns {violin | number} + * @memberof violin + * @property + * by default pointRadius = 2 + */ + violin.pointRadius = function(_) { return arguments.length ? (pointRadius = _, violin) : pointRadius; }; + /** + * Gets / sets the pointStrokeWidth + * (see {@link violin#pointStrokeWidth}) + * @param {number} [_=none] + * @returns {violin | number} + * @memberof violin + * @property + * by default pointStrokeWidth = 2 + */ + violin.pointStrokeWidth = function(_) { return arguments.length ? (pointStrokeWidth = _, violin) : pointStrokeWidth; }; - // console.utils.con.log(set, intersection) - t.classed(intersection, true); - t.classed(set, true); + /** + * Gets / sets the backgroundFill + * (see {@link violin#backgroundFill}) + * @param {string} [_=none] + * @returns {violin | string} + * @memberof violin + * @property + * by default backgroundFill = 'transparent' + */ + violin.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, violin) : backgroundFill; }; + /** + * Gets / sets the namespace + * (see {@link violin#namespace}) + * @param {string} [_=none] + * @returns {violin | string} + * @memberof violin + * @property + * by default namespace = 'd3sm-violin' + */ + violin.namespace = function(_) { return arguments.length ? (namespace = _, violin) : namespace; }; + /** + * Gets / sets the objectClass + * (see {@link violin#objectClass}) + * @param {string} [_=none] + * @returns {violin | string} + * @memberof violin + * @property + * by default objectClass = 'tick-group' + */ + violin.objectClass = function(_) { return arguments.length ? (objectClass = _, violin) : objectClass; }; - var c = utils$1.sel.safeSelect(t, 'circle', utils$1.str.hypenate(objectClass,'circle')); - c.attr('cx', xObjectSize / 2) - .attr('cy', yObjectSize / 2 ) - .attr('r', radius == undefined ? Math.min(xObjectSize, yObjectSize) / 2 : radius) - .attr('fill', intersection.includes(set) ? "black": 'rgb(233,233,233)') - .attr('stroke', "black") - .attr("in-intersection", intersection.includes(set)); - }); + /** + * Gets / sets the transitionDuration + * (see {@link violin#transitionDuration}) + * @param {number} [_=none] + * @returns {violin | number} + * @memberof violin + * @property + * by default transitionDuration = 1000 + */ + violin.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, violin) : transitionDuration; }; + /** + * Gets / sets the easeFunc + * (see {@link violin#easeFunc}) + * @param {d3.ease} [_=none] + * @returns {violin | d3.ease} + * @memberof violin + * @property + * by default easeFunc = d3.easeExp + */ + violin.easeFunc = function(_) { return arguments.length ? (easeFunc = _, violin) : easeFunc; }; - } + /** + * Gets / sets the quartileKey + * (see {@link violin#quartileKey}) + * @param {string} [_=none] + * @returns {violin | string} + * @memberof violin + * @property + * by default quartileKey = "utils.math.quartiles" + */ + violin.quartileKey = function(_) { return arguments.length ? (quartileKey = _, violin) : quartileKey; }; + /** + * Gets / sets the quartileKeys + * (see {@link violin#quartileKeys}) + * @param {string[]} [_=none] + * @returns {violin | string[]} + * @memberof violin + * @property + * by default quartileKeys = ["Q0","Q1","Q2","Q3","Q4"] + */ + violin.quartileKeys = function(_) { return arguments.length ? (quartileKeys = _, violin) : quartileKeys; }; - function intersectionTotals() { - var totals = {}; - // intersectionValues.sort(function(a, b){ return intersectionKeySortingFunction(a, b) }) - intersectionValues.map(function(k, i){ totals[k] = {'total': 0}; }); - cellKeys.map(function(k, i){ - var e = elementExtractor(k, i); - if (totals[intersectionExtractor(k, i)]['total'] == 0) { - if (Array.isArray(e)) { - totals[intersectionExtractor(k, i)]['total']+= e.length; - totals[intersectionExtractor(k, i)]['values'] = e; - } else { - totals[intersectionExtractor(k, i)]['total']+= e; - } + /** + * Gets / sets the violinKeys + * (see {@link violin#violinKeys}) + * @param {string[]} [_=none] + * @returns {violin | string[]} + * @memberof violin + * @property + * by default violinKeys = undefined + */ + violin.violinKeys = function(_) { return arguments.length ? (violinKeys = _, violin) : violinKeys; }; + /** + * Gets / sets the violinValues + * (see {@link violin#violinValues}) + * @param {Object[]} [_=none] + * @returns {violin | Object[]} + * @memberof violin + * @property + * by default violinValues = undefined + */ + violin.violinValues = function(_) { return arguments.length ? (violinValues = _, violin) : violinValues; }; - } - }); - return totals - } + /** + * Gets / sets the objectSize + * (see {@link violin#objectSize}) + * @param {number} [_=none] + * @returns {violin | number} + * @memberof violin + * @property + * by default objectSize = undefined + */ + violin.objectSize = function(_) { return arguments.length ? (objectSize = _, violin) : objectSize; }; + /** + * Gets / sets the spacerSize + * (see {@link violin#spacerSize}) + * @param {number} [_=none] + * @returns {violin | number} + * @memberof violin + * @property + * by default spacerSize = undefined + */ + violin.spacerSize = function(_) { return arguments.length ? (spacerSize = _, violin) : spacerSize; }; + /** + * Gets / sets the tooltip + * (see {@link violin#tooltip}) + * @param {tooltip} [_=none] + * @returns {violin | tooltip} + * @memberof violin + * @property + * by default tooltip = tooltip() + */ + violin.tooltip = function(_) { return arguments.length ? (tooltip$$1 = _, violin) : tooltip$$1; }; + /** + * Gets / sets the pointsTooltip + * (see {@link violin#pointsTooltip}) + * @param {tooltip} [_=none] + * @returns {violin | tooltip} + * @memberof violin + * @property + * by default pointsTooltip = tooltip() + */ + violin.pointsTooltip = function(_) { return arguments.length ? (pointsTooltip = _, violin) : pointsTooltip; }; - function setTotals(){ - var totals = {}; - // intersectionValues.sort(function(a, b){ return intersectionKeySortingFunction(a, b) }) - setValues.map(function(k, i){ totals[k] = {'total': 0}; }); - cellKeys.map(function(k, i){ - var e = elementExtractor(k, i); - if (Array.isArray(e)) { - totals[setExtractor(k, i)]['total']+= e.length; - } else { - totals[setExtractor(k, i)]['total']+= e; - } - }); - return totals - } - upset.intersectionTotals = intersectionTotals; - upset.setTotals = setTotals; - return upset - } + function violin () { + // for convenience in handling orientation specific values + var horizontalQ = (orient == 'horizontal') ? true : false; - let charts = { - scatter, bar, bubble: bubbleHeatmap, heatmap, violin, neededViolinValues, upset - }; + // background cliping rectangle + var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; + var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); - function categoricLegend( selection ) { - var - categories, + // if grouping is undefined sort violinKeys by sortingFunction + var ordered = (grouping == undefined) ? d3.keys(data).sort(sortingFunction) : grouping; - /** - * Data to plot. Assumed to be a object, where each key corresponds to a legend - * (see {@link legend#data}) - * @param {Object} [data=undefined] - * @memberof legend# - * @property - */ - data, - /** - * Which direction to render the bars in - * (see {@link legend#orient}) - * @param {number} [orient='horizontal'] - * @memberof legend# - * @property - */ - orient='horizontal', - /** - * Amount of horizontal space (in pixels) avaible to render the legend in - * (see {@link legend#spaceX}) - * @param {number} [spaceX=undefined] - * @memberof legend# - * @property - */ - spaceX, - /** - * Amount of vertical space (in pixels) avaible to render the legend in - * (see {@link legend.spaceY}) - * @param {number} [spaceY=undefined] - * @memberof legend# - * @property - */ - spaceY, + // console.log(ordered) - /** - * Whether or not to allow legend to render elements pass the main spatial dimension - * given the orientation (see {@link legend#orient}), where {@link legend#orient}="horizontal" - * the main dimension is {@link legend#spaceX} and where {@link legend#orient}="vertical" - * the main dimension is {@link legend#spaceY} - * @param {boolean} [overflowQ=false] - * @memberof legend# - * @property - */ - overflowQ = false, + violinKeys = utils.arr.flatten(ordered); - /** - * An array - putatively of other arrays - depicting how bars should be arranged - * @param {Array[]} [grouping=undefined] - * @memberof legend# - * @property - */ - grouping, - /** - * How to get the value of the legend - * @param {function} [valueExtractor=function(key, index) { return data[key] }] - * @memberof legend# - * @property - */ - valueExtractor = function(key, index) { return data[key] }, - /** - * How to sort the bars - if {@link bar#grouping} is not provided. - * @param {function} [sortingFunction=function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}] - * @memberof bar# - * @property - */ - sortingFunction = function(keyA, keyB) {return d3.ascending(keyA, keyB)}, - /** - * Default space for the spacer (percentage) of main dimension given the orientation - * (see {@link legend#orient}), where {@link legend#orient}="horizontal" - * the main dimension is {@link legend#spaceX} and where {@link legend#orient}="vertical" - * the main dimension is {@link legend#spaceY} between bars - * @param {number} [objectSpacer=0.05] - * @memberof legend# + var calcValues = neededViolinValues() + .horizontalQ(horizontalQ) + .quartileKeys(quartileKeys) + .violinPointsExtractor(violinPointsExtractor) + .violinPointValueExtractor(violinPointValueExtractor); + + + + // augment valus + violinValues = {}; + violinKeys.map(function(vk, i){ + let v = calcValues(vk, data); + violinValues[vk] = v; + }); + + var numberOfObjects = violinKeys.length; + + var min = [].concat(...violinKeys.map(function(k, i){return data[k].quartiles[quartileKeys[0]]})); + var max = [].concat(...violinKeys.map(function(k, i){return data[k].quartiles[quartileKeys[quartileKeys.length - 1]]})); + var extent = [Math.min(...min) - domainPadding, Math.max(...max) + domainPadding]; + // console.log(extent, violinValues, ordered) + + // set the scale + scale.domain(extent).range(horizontalQ ? [0,spaceY] : [0, spaceX]); + var space = horizontalQ ? spaceX : spaceY; + // calculate object size + objectSize = utils.math.calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ); + // calculate spacer size if needed + spacerSize = utils.math.calculateWidthOfSpacer(ordered, space, objectSize, numberOfObjects, objectSpacer, overflowQ); + // make the nested groups + var spacerFunction = groupingSpacer() + .horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects) + .objectClass(objectClass).objectSize(objectSize).spacerSize(spacerSize) + .transitionDuration(transitionDuration).easeFunc(easeFunc) + .namespace(namespace); + + // move stuff + spacerFunction(container, ordered, 0); + // console.log(violinKeys, ordered, container.selectAll('g:not(.to-remove).'+objectClass).nodes()) + + // for color function + var parentIndexArray = []; + container.selectAll('g:not(.to-remove).'+objectClass) + .each(function(d, i){if (utils.arr.hasQ(violinKeys, d)){ parentIndexArray.push(Number(d3.select(this).attr('parent-index')));}}); + + // update color function + colorFunction$$1 = colorFunction$$1.colorBy() == 'index' + ? colorFunction$$1.dataExtent([0, Math.max(...parentIndexArray)]) + : colorFunction$$1.dataExtent(extent); + + /* violiin specific needs */ + + + var frequencyMax = Math.max(...[].concat(...violinKeys.map(function(k, i){return d3.max(data[k].frequencies)}))); + var vScale = d3.scaleLinear().domain([0, frequencyMax]).range([0, objectSize / 2]); + + var lArea = d3.line() + .x(function(d, i){ return horizontalQ ? -vScale(d.x) : scale(d.x)}) + .y(function(d, i){ return horizontalQ ? scale(extent[1]) - scale(d.y) : -vScale(d.y)}) + .curve(d3.curveBasis); + var rArea = d3.line() + .x(function(d, i){ return horizontalQ ? vScale(d.x) : scale(d.x)}) + .y(function(d, i){ return horizontalQ ? scale(extent[1]) - scale(d.y) : vScale(d.y)}) + .curve(d3.curveBasis); + + + + + + + container.selectAll('g:not(.to-remove).'+objectClass).each(function(key, i){ + var t = d3.select(this), + currentData = data[key]; + // needed because bug in selecting .to-remove + if (!utils.arr.hasQ(violinKeys, key)) {return} + var + i = t.attr('parent-index') == undefined ? i : t.attr('parent-index'), + fillColor = colorFunction$$1(key, currentData, i, 'fill'), // prevent duplicate computation + strokeColor = colorFunction$$1(key, currentData, i, 'stroke'), + area = utils.sel.safeSelect(t, 'g', 'area'), + la = utils.sel.safeSelect(area, 'path', 'left'), + ra = utils.sel.safeSelect(area, 'path', 'right'), + quarts = utils.sel.safeSelect(t, 'g', 'quarts'), + lq3 = utils.sel.safeSelect(quarts, 'line', 'q3'), + lq1 = utils.sel.safeSelect(quarts, 'line', 'q1'), + q3 = currentData.quartiles[quartileKeys[3]], + q2 = currentData.quartiles[quartileKeys[2]], + q1 = currentData.quartiles[quartileKeys[1]]; + + t.attr('transform', horizontalQ ? 'translate('+objectSize / 2+',0)' : 'translate(0,'+objectSize / 2+')' ); + // draw curve + la.transition().duration(transitionDuration).attr('d', function(dd, ii){ return lArea(currentData.contour)}) + .attr('fill', fillColor) + .attr('stroke', strokeColor) + .attr('stroke-width', objectStrokeWidth); + + ra.transition().duration(transitionDuration).attr('d', function(dd, ii){ return rArea(currentData.contour)}) + .attr('fill', fillColor) + .attr('stroke', strokeColor) + .attr('stroke-width', objectStrokeWidth); + + area.node().addEventListener('mouseover', function(dd, ii){ + container.selectAll('g.'+objectClass).style('opacity', 0.2); + t.style('opacity', 1); + la.attr('stroke-width',objectStrokeWidth*2); + ra.attr('stroke-width',objectStrokeWidth*2); + }); + area.node().addEventListener('mouseout', function(dd, ii){ + container.selectAll('g.'+objectClass).style('opacity', 1); + la.attr('stroke-width',objectStrokeWidth); + ra.attr('stroke-width',objectStrokeWidth); + }); + + if (pointsQ) { + var ptsContainer = utils.sel.safeSelect(t, 'g', 'points'); + var pts = ptsContainer.selectAll('.point').data(currentData.pointKeys); + pts.on('mouseover', null); + + + var ptsExit = pts.exit().transition().ease(easeFunc).duration(transitionDuration) + .attr('r', 0) + .attr('cy', horizontalQ ? scale(extent[1]) - scale(q2) : vScale(0)) + .attr('cx', horizontalQ ? vScale(0) : scale(q2)).remove(); + + var ptsEnter = pts.enter().append('circle').attr('class', 'point').attr('r', 0) + .attr('cx', horizontalQ ? 0 : scale(q2)) + .attr('cy', horizontalQ ? scale(q2) : 0); + + pts = pts.merge(ptsEnter); + + // console.log(pointsTooltip.header()) + + var pTTips = tooltip().selection(pts).data(violinPointsExtractor(key, currentData)) + .header(pointsTooltip.header()) + .keys(pointsTooltip.keys()) + .values(pointsTooltip.values()); + + pTTips(); + + var pMin = d3.min(currentData.pointValues), pMax = d3.max(currentData.pointValues); + + + + pts.transition().duration(transitionDuration).ease(easeFunc).attr('r', pointRadius) + .attr('cy', function(pointKey, ii){ + var dd = currentData.pointValues[ii]; + if (horizontalQ) { return scale(extent[1]) - scale(dd) } + var j = utils.arr.whichBin(currentData.binned, dd); + var r = Math.random(); + var n = vScale(r * currentData.frequencies[j] * 0.5); + var k = Math.random() > 0.5 ? n : -n; + return k + }) + .attr('cx', function(pointKey, ii){ + var dd = currentData.pointValues[ii]; + if (horizontalQ) { + var j = utils.arr.whichBin(currentData.binned, dd); + var r = Math.random(); + var n = vScale(r * currentData.frequencies[j] * 0.5); + var k = Math.random() > 0.5 ? n : -n; + return k + } + return scale(dd) + }) + .attr('stroke', function(dd, ii) { var dd = currentData.pointValues[ii]; return pointColorFunc(dd, 'stroke', strokeColor, pMin, pMax) }) + .attr('fill' , function(dd, ii) { var dd = currentData.pointValues[ii]; return pointColorFunc(dd, 'fill' , strokeColor, pMin, pMax) }) + .attr('stroke-width', pointStrokeWidth); + + ptsContainer.selectAll('circle.point').on('mouseover', function(dd, ii){ + container.selectAll('g.'+objectClass).style('opacity', 0.2); + t.style('opacity', 1); + la.attr('stroke-width',objectStrokeWidth*2); + ra.attr('stroke-width',objectStrokeWidth*2); + + container.selectAll('.point').style('opacity', 0.2); + d3.select(this).style('opacity', 1).attr('r', pointRadius * 2).attr('stroke-width',pointStrokeWidth*2); + }); + ptsContainer.selectAll('circle.point').on('mouseout', function(dd, ii){ + var e = document.createEvent('SVGEvents'); + e.initEvent('mouseout',true,true); + area.node().dispatchEvent(e); + + container.selectAll('.point').style('opacity', 1); + d3.select(this).attr('stroke-width', pointStrokeWidth).attr('r', pointRadius); + }); + } + else { + cV.selectAll('.point') + .transition().duration(transitionDuration).ease(easeFunc) + .attr('r', 0) + .attr('cy', horizontalQ ? scale(extent[1]) - scale(q2) : vScale(0)) + .attr('cx', horizontalQ ? vScale(0) : scale(q2)) + .remove(); + } + + + }); + + + tooltip$$1.selection(container.selectAll('g:not(.to-remove).'+objectClass + ' .area')); + if (tooltip$$1.data() == undefined) {tooltip$$1.data(data);} + tooltip$$1(); + if (tooltip$$1.values() == undefined) { + tooltip$$1.values([ + function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] }, + function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] }, + function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] }, + function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] }, + function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] } + ]); + + } + + } + + return violin + } + + + + + + /** + * Produces the function which manipulates the violin data to have values needed + * for rendering the violins as svg. + * @returns {function} calculateViolinValues + * @namespace neededViolinValues + */ + function neededViolinValues() { + var + /** + * Whether or not the orientation of the violins are horizontal + * (see {@link violin#orient}) + * @param {Object} [horizontalQ=true] + * @memberof neededViolinValues# * @property */ - objectSpacer = 0.05, + horizontalQ = true, /** - * The minimum size that an object can be - * @param {number} [minObjectSize=50] - * @memberof legend# + * Keys to be put into the utils.math.quartiles if they need to be calculated. + * (see {@link violin#quartileKeys}) + * @param {Object} [quartileKeys=['Q0', 'Q1', 'Q2', 'Q3', 'Q4']] + * @memberof neededViolinValues# * @property */ - minObjectSize = 10, + quartileKeys = ['Q0', 'Q1', 'Q2', 'Q3', 'Q4'], /** - * The maximum size that an object can be - * @param {number} [maxObjectSize=100] - * @memberof legend# + * Function which given the key of the violin and that key's associated value + * returns the object consiting of the points of the violin + * (see {@link violin#violinPointsExtractor}) + * @param {Object} [violinPointsExtractor=function(violinKey, violinData) { return violinData.points }] + * @memberof neededViolinValues# * @property */ - maxObjectSize = 100, - + violinPointsExtractor, /** - * The stroke width of the bars - * @param {number} [barStrokeWidth=2] - * @memberof legend# + * Function which given the key of the current point and the object of points for the + * violin, returns the numerical value of the point + * (see {@link violin#violinPointValueExtractor}) + * @param {Object} [violinPointValueExtractor=function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value }] + * @memberof neededViolinValues# * @property */ - bubbleStrokeWidth = 2, - /** - * Instance of ColorFunction - * @param {function} [colorFunction = colorFunction()] - * @memberof legend# - * @property - */ - colorFunction$$1 = colorFunction(), + violinPointValueExtractor; + /** - * Color of the background - * @param {string} [backgroundFill="transparent"] - * @memberof legend# - * @property - */ - backgroundFill = 'transparent', + * Gets / sets the horizontalQ + * (see {@link violin#orient}) + * @param {boolean} [_=none] + * @returns {calculateViolinValues | boolean} + * @memberof calculateViolinValues + * @property + * + * by default horizontalQ = true + */ + calculateViolinValues.horizontalQ = function(_) { return arguments.length ? (horizontalQ=_, calculateViolinValues) : horizontalQ }; /** - * Namespace for all items made by this instance of legend - * @param {string} [namespace="d3sm-legend"] - * @memberof legend# - * @property - */ - namespace = 'd3sm-legend', + * Gets / sets the quartileKeys + * (see {@link violin#quartileKeys}) + * @param {string[]} [_=none] + * @returns {calculateViolinValues | string[]} + * @memberof calculateViolinValues + * @property + * + * by default quartileKeys = ["Q0","Q1","Q2","Q3","Q4"] + */ + calculateViolinValues.quartileKeys = function(_) { return arguments.length ? (quartileKeys=_, calculateViolinValues) : quartileKeys }; /** - * Class name for legend container ( element) - * @param {string} [objectClass="legend"] - * @memberof legend# + * Gets / sets the violinPointsExtractor + * (see {@link violin#violinPointsExtractor}) + * @param {function} [_=none] + * @returns {calculateViolinValues | function} + * @memberof calculateViolinValues + * @property + * + * by default violinPointsExtractor = function(violinKey, violinData) { return violinData.points } + */ + calculateViolinValues.violinPointsExtractor = function(_) { return arguments.length ? (violinPointsExtractor=_, calculateViolinValues) : violinPointsExtractor }; + /** + * Gets / sets the violinPointValueExtractor + * (see {@link violin#violinPointValueExtractor}) + * @param {function} [_=none] + * @returns {calculateViolinValues | function} + * @memberof calculateViolinValues + * @property + * + * by default violinPointValueExtractor = function(violinPointKey, violinPointData) { return violinPointData[violinPointKey].value } + */ + calculateViolinValues.violinPointValueExtractor = function(_) { return arguments.length ? (violinPointValueExtractor=_, calculateViolinValues) : violinPointValueExtractor }; + + + /** + * The function produced by neededViolinValues. + * + * Adds the data need to render the violin as an svg + * @param {string} violinKey the key of the current violin + * @param {object} data the object consisting of violinKey - values (violin data) pairs + * @returns {none} this function manipulates the passed data object adding the following keys + * + * data[violinKey].binned // the binned values of the points + * + * data[violinKey].frequencies // the list consisting of the length of each bin + * + * data[violinKey].contour // the points depicting the contour of 1/2 of the violin + * + * data[violinKey].utils.math.quartiles // the utils.math.quartiles of the points' values + * + * data[violinKey].pointKeys // the keys associated with the points + * + * data[violinKey].pointValues // the numerical values of the points + * + * @memberof neededViolinValues# * @property */ - objectClass = 'legend', + function calculateViolinValues(violinKey, data) { + // data for the current violin + var violinData = data[violinKey]; + // the object of points + var violinPoints = violinPointsExtractor(violinKey, violinData); + // + var violinPointsKeys = d3.keys(violinPoints); + // the numerical values of those points + var violinPointsValues = violinPointsKeys.map(function(pk, i){return violinPointValueExtractor(pk, violinPoints)}); + + // utils.math.quartiles of those points + var pointQuartiles = utils.math.quartiles(violinPointsValues, quartileKeys); + + // binned points + var binned = d3.histogram()(violinPointsValues); + // length of bins + var frequencies = binned.map(bin=>bin.length); + // min and max countour points for nice drawings + var minContourPoint = horizontalQ ? {x: 0, y: d3.min(violinPointsValues)} : {x: d3.min(violinPointsValues), y: 0}; + var maxContourPoint = horizontalQ ? {x: 0, y: d3.max(violinPointsValues)} : {x: d3.max(violinPointsValues), y: 0}; + var violinContourPoints = binned.map(function(bin, i) { + return horizontalQ + ? {y: (bin.length) ? d3.median(bin): d3.median([bin.x0, bin.x1]), x: frequencies[i]} + : {x: (bin.length) ? d3.median(bin): d3.median([bin.x0, bin.x1]), y: frequencies[i]} + }); + // points along which to draw the violin shpe + violinContourPoints = [minContourPoint].concat(violinContourPoints).concat([maxContourPoint]); + + // set data + violinData.binned = binned; + violinData.frequencies = frequencies; + violinData.contour = violinContourPoints; + violinData.quartiles = pointQuartiles; + violinData.pointKeys = violinPointsKeys; + violinData.pointValues = violinPointsValues; + return violinData + } + + return calculateViolinValues + } + + function upset ( selection ) { + var + data, + orient='horizontal', + spaceX, + spaceY, + overflowQ=false, + minObjectSize=20, + maxObjectSize=50, + circleStrokeWidth=2, + // colorFunction= + backgroundFill = 'transparent', + namespace='d3sm-upset', + objectClass = 'upset', + + transitionDuration = 1000, + easeFunc = d3.easeExp, + + setKey = "set", + intersectionKey = "intersection", + elementsKey = "elements", + + setExtractor = function(key, i) {return data[key][setKey]}, + intersectionExtractor = function(key, i) {return data[key][intersectionKey]}, + elementExtractor = function(key, i) {return data[key][elementsKey]}, + + cellKeys, + setValues, + intersectionValues, + xObjectSpacer = 0.05, + yObjectSpacer = 0.05, + radius, + + // listDelim = ';' + + yObjectSize, + ySpacerSize, + xObjectSize, + xSpacerSize, + + setKeySortingFunction = function(a, b) { return setValues.indexOf(setExtractor(a)) - setValues.indexOf(setExtractor(b)) }, + intersectionKeySortingFunction = function(a, b) { return intersectionValues.indexOf(intersectionExtractor(a)) - intersectionValues.indexOf(intersectionExtractor(b)) }; + + + upset.selection = function(_) { return arguments.length ? (selection = _, upset) : selection; }; + upset.data = function(_) { return arguments.length ? (data = _, upset) : data; }; + upset.orient = function(_) { return arguments.length ? (orient = _, upset) : orient; }; + upset.spaceX = function(_) { return arguments.length ? (spaceX = _, upset) : spaceX; }; + upset.spaceY = function(_) { return arguments.length ? (spaceY = _, upset) : spaceY; }; + upset.overflowQ = function(_) { return arguments.length ? (overflowQ = _, upset) : overflowQ; }; + upset.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, upset) : minObjectSize; }; + upset.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, upset) : maxObjectSize; }; + upset.circleStrokeWidth = function(_) { return arguments.length ? (circleStrokeWidth = _, upset) : circleStrokeWidth; }; + upset.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, upset) : backgroundFill; }; + upset.namespace = function(_) { return arguments.length ? (namespace = _, upset) : namespace; }; + upset.objectClass = function(_) { return arguments.length ? (objectClass = _, upset) : objectClass; }; + upset.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, upset) : transitionDuration; }; + upset.easeFunc = function(_) { return arguments.length ? (easeFunc = _, upset) : easeFunc; }; + upset.cellKeys = function(_) { return arguments.length ? (cellKeys = _, upset) : cellKeys; }; + upset.setValues = function(_) { return arguments.length ? (setValues = _, upset) : setValues; }; + upset.intersectionValues = function(_) { return arguments.length ? (intersectionValues = _, upset) : intersectionValues; }; + upset.xObjectSpacer = function(_) { return arguments.length ? (xObjectSpacer = _, upset) : xObjectSpacer; }; + upset.yObjectSpacer = function(_) { return arguments.length ? (yObjectSpacer = _, upset) : yObjectSpacer; }; + upset.radius = function(_) { return arguments.length ? (radius = _, upset) : radius; }; + upset.setExtractor = function(_) { return arguments.length ? (setExtractor = _, upset) : setExtractor; }; + upset.intersectionExtractor = function(_) { return arguments.length ? (intersectionExtractor = _, upset) : intersectionExtractor; }; + upset.elementExtractor = function(_) { return arguments.length ? (elementExtractor = _, upset) : elementExtractor; }; + upset.setKeySortingFunction = function(_) { return arguments.length ? (setKeySortingFunction = _, upset) : setKeySortingFunction; }; + upset.intersectionKeySortingFunction = function(_) { return arguments.length ? (intersectionKeySortingFunction = _, upset) : intersectionKeySortingFunction; }; + + upset.yObjectSize = function(_) { return arguments.length ? (yObjectSize = _, upset) : yObjectSize; }; + upset.ySpacerSize = function(_) { return arguments.length ? (ySpacerSize = _, upset) : ySpacerSize; }; + upset.xObjectSize = function(_) { return arguments.length ? (xObjectSize = _, upset) : xObjectSize; }; + upset.xSpacerSize = function(_) { return arguments.length ? (xSpacerSize = _, upset) : xSpacerSize; }; + + function upset() { + // for convenience in handling orientation specific values + var horizontalQ = (orient == 'horizontal') ? true : false; + var verticalQ = !horizontalQ; + + // background cliping rectangle + var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; + var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); + + + cellKeys = d3.keys(data); + setValues = utils.arr.unique(cellKeys.map(setExtractor)).sort(); + intersectionValues = utils.arr.unique(cellKeys.map(intersectionExtractor)).sort().sort(function(a, b){ + return a.split(';').length - b.split(';').length + }); + + if (!horizontalQ) { + cellKeys.sort(function(a, b){ return setKeySortingFunction(a, b) || intersectionKeySortingFunction(a, b) }); + } else { + cellKeys.sort(function(a, b){ return intersectionKeySortingFunction(a, b) || setKeySortingFunction(a, b) }); + } + + + + + var + xValues = horizontalQ ? intersectionValues : setValues, + yValues = horizontalQ ? setValues : intersectionValues, + xDim = horizontalQ ? xValues.length : yValues.length, + yDim = horizontalQ ? yValues.length : xValues.length; + + // console.utils.con.log(xValues, yValues) + + + xObjectSize = utils.math.calculateWidthOfObject(spaceX, xDim, minObjectSize, maxObjectSize, xObjectSpacer, overflowQ); + yObjectSize = utils.math.calculateWidthOfObject(spaceY, yDim, minObjectSize, maxObjectSize, yObjectSpacer, overflowQ); + xSpacerSize = utils.math.calculateWidthOfSpacer(xValues, spaceX, xObjectSize, xDim, xObjectSpacer, overflowQ); + ySpacerSize = utils.math.calculateWidthOfSpacer(yValues, spaceY, yObjectSize, yDim, yObjectSpacer, overflowQ); + + var ySpacer = groupingSpacer() + .horizontalQ(false) + .moveby('category').numberOfObjects(yDim) + .objectSize(yObjectSize).spacerSize(ySpacerSize) + .transitionDuration(transitionDuration).easeFunc(easeFunc); + + var xSpacer = groupingSpacer() + .horizontalQ(true) + .moveby('category').numberOfObjects(xDim) + .objectClass(objectClass) + .objectSize(xObjectSize).spacerSize(xSpacerSize) + .transitionDuration(transitionDuration).easeFunc(easeFunc); + + + + if (verticalQ) { + xSpacer.objectClass(objectClass); + ySpacer.namespace('across').objectClass(utils.str.hypenate(objectClass, 'across')); + + ySpacer(container, yValues, 0); + container.selectAll('g.'+utils.str.hypenate(objectClass, 'across')) + .each(function(d, i){ xSpacer(d3.select(this), xValues, 0); }); + } else { + xSpacer.namespace('across').objectClass(utils.str.hypenate(objectClass, 'across')); + ySpacer.objectClass(objectClass); + + xSpacer(container, xValues, 0); + container.selectAll('g.'+utils.str.hypenate(objectClass, 'across')) + .each(function(d, i){ ySpacer(d3.select(this), yValues, 0); }); + } + + + var cells = container.selectAll('g:not(.to-remove).'+objectClass); + var lookup = {}; + cellKeys.map(function(k, i){ + lookup[setExtractor(k)+'::'+intersectionExtractor(k)] = k; + }); + + // var positionedCellKeys = [] + // for (var i = 0; i < setValues.length; i++) { + // for (var j = 0; j < intersectionValues.length; j++) { + // var lookupValue = lookup[setValues[j]+"::"+intersectionValues[i]] + // if (lookupValue == undefined) { + // positionedCellKeys.push(undefined) + // } else { + // positionedCellKeys.push(lookupValue) + // } + // console.utils.con.log(i, j, lookupValue) + // } + // } + // + // // console.utils.con.log(positionedCellKeys) + // + // cells.data(positionedCellKeys); + + + cells.data(cellKeys); + + cells.each(function(key, i) { + var t = d3.select(this); + if (key == undefined) {return } + var + currentData = data[key]; + // console.utils.con.log(key, currentData) + var + set = setExtractor(key, i), + intersection = intersectionExtractor(key, i); + + // console.utils.con.log(set, intersection) + + t.classed(intersection, true); + t.classed(set, true); + + var c = utils.sel.safeSelect(t, 'circle', utils.str.hypenate(objectClass,'circle')); + c.attr('cx', xObjectSize / 2) + .attr('cy', yObjectSize / 2 ) + .attr('r', radius == undefined ? Math.min(xObjectSize, yObjectSize) / 2 : radius) + .attr('fill', intersection.includes(set) ? "black": 'rgb(233,233,233)') + .attr('stroke', "black") + .attr("in-intersection", intersection.includes(set)); + }); + + + + } + + function intersectionTotals() { + var totals = {}; + // intersectionValues.sort(function(a, b){ return intersectionKeySortingFunction(a, b) }) + + intersectionValues.map(function(k, i){ totals[k] = {'total': 0}; }); + cellKeys.map(function(k, i){ + var e = elementExtractor(k, i); + if (totals[intersectionExtractor(k, i)]['total'] == 0) { + if (Array.isArray(e)) { + totals[intersectionExtractor(k, i)]['total']+= e.length; + totals[intersectionExtractor(k, i)]['values'] = e; + } else { + totals[intersectionExtractor(k, i)]['total']+= e; + } + + } + }); + return totals + } + + function setTotals(){ + var totals = {}; + // intersectionValues.sort(function(a, b){ return intersectionKeySortingFunction(a, b) }) + + setValues.map(function(k, i){ totals[k] = {'total': 0}; }); + + cellKeys.map(function(k, i){ + var e = elementExtractor(k, i); + if (Array.isArray(e)) { + totals[setExtractor(k, i)]['total']+= e.length; + } else { + totals[setExtractor(k, i)]['total']+= e; + } + }); + return totals + } + + upset.intersectionTotals = intersectionTotals; + upset.setTotals = setTotals; + + return upset + } + + let charts = { + scatter, bar, bubble: bubbleHeatmap, boxwhisker, heatmap, violin, neededViolinValues, upset + }; + + function categoricLegend( selection ) { + var + categories, + + /** + * Data to plot. Assumed to be a object, where each key corresponds to a legend + * (see {@link legend#data}) + * @param {Object} [data=undefined] + * @memberof legend# + * @property + */ + data, + /** + * Which direction to render the bars in + * (see {@link legend#orient}) + * @param {number} [orient='horizontal'] + * @memberof legend# + * @property + */ + orient='horizontal', + /** + * Amount of horizontal space (in pixels) avaible to render the legend in + * (see {@link legend#spaceX}) + * @param {number} [spaceX=undefined] + * @memberof legend# + * @property + */ + spaceX, + /** + * Amount of vertical space (in pixels) avaible to render the legend in + * (see {@link legend.spaceY}) + * @param {number} [spaceY=undefined] + * @memberof legend# + * @property + */ + spaceY, + + /** + * Whether or not to allow legend to render elements pass the main spatial dimension + * given the orientation (see {@link legend#orient}), where {@link legend#orient}="horizontal" + * the main dimension is {@link legend#spaceX} and where {@link legend#orient}="vertical" + * the main dimension is {@link legend#spaceY} + * @param {boolean} [overflowQ=false] + * @memberof legend# + * @property + */ + overflowQ = false, + + /** + * An array - putatively of other arrays - depicting how bars should be arranged + * @param {Array[]} [grouping=undefined] + * @memberof legend# + * @property + */ + grouping, + + /** + * How to get the value of the legend + * @param {function} [valueExtractor=function(key, index) { return data[key] }] + * @memberof legend# + * @property + */ + valueExtractor = function(key, index) { return data[key] }, + /** + * How to sort the bars - if {@link bar#grouping} is not provided. + * @param {function} [sortingFunction=function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}] + * @memberof bar# + * @property + */ + sortingFunction = function(keyA, keyB) {return d3.ascending(keyA, keyB)}, + /** + * Default space for the spacer (percentage) of main dimension given the orientation + * (see {@link legend#orient}), where {@link legend#orient}="horizontal" + * the main dimension is {@link legend#spaceX} and where {@link legend#orient}="vertical" + * the main dimension is {@link legend#spaceY} between bars + * @param {number} [objectSpacer=0.05] + * @memberof legend# + * @property + */ + objectSpacer = 0.05, + /** + * The minimum size that an object can be + * @param {number} [minObjectSize=50] + * @memberof legend# + * @property + */ + minObjectSize = 10, + /** + * The maximum size that an object can be + * @param {number} [maxObjectSize=100] + * @memberof legend# + * @property + */ + maxObjectSize = 100, + + /** + * The stroke width of the bars + * @param {number} [barStrokeWidth=2] + * @memberof legend# + * @property + */ + bubbleStrokeWidth = 2, + /** + * Instance of ColorFunction + * @param {function} [colorFunction = colorFunction()] + * @memberof legend# + * @property + */ + colorFunction$$1 = colorFunction(), + + /** + * Color of the background + * @param {string} [backgroundFill="transparent"] + * @memberof legend# + * @property + */ + backgroundFill = 'transparent', + /** + * Namespace for all items made by this instance of legend + * @param {string} [namespace="d3sm-legend"] + * @memberof legend# + * @property + */ + namespace = 'd3sm-legend', + /** + * Class name for legend container ( element) + * @param {string} [objectClass="legend"] + * @memberof legend# + * @property + */ + objectClass = 'legend', + + /** + * Duration of all transitions of this element + * @param {number} [transitionDuration=1000] + * @memberof legend# + * @property + */ + transitionDuration = 1000, + /** + * Easing function for transitions + * @param {d3.ease} [easeFunc=d3.easeExp] + * @memberof legend# + * @property + */ + easeFunc = d3.easeExp; + + + legend.categories = function(_) { return arguments.length ? (categories=_, legend) : categories }; + + /** + * Gets or sets the selection in which items are manipulated + * @param {d3.selection} [_=none] + * @returns {legend | d3.selection} + * @memberof legend + * @property + * by default selection = selection + */ + legend.selection = function(_) { return arguments.length ? (selection = _, legend) : selection; }; + /** + * Gets or sets the data + * (see {@link legend#data}) + * @param {number} [_=none] + * @returns {legend | object} + * @memberof legend + * @property + */ + legend.data = function(_) { return arguments.length ? (data = _, legend) : data; }; + /** + * Gets or sets the orient of the bars + * (see {@link legend#orient}) + * @param {number} [_=none] + * @returns {legend | object} + * @memberof legend + * @property + */ + legend.orient = function(_) { return arguments.length ? (orient = _, legend) : orient; }; + /** + * Gets or sets the amount of horizontal space in which items are manipulated + * (see {@link legend#spaceX}) + * @param {number} [_=none] should be a number > 0 + * @returns {legend | number} + * @memberof legend + * @property + * by default spaceX = undefined + */ + legend.spaceX = function(_) { return arguments.length ? (spaceX = _, legend) : spaceX; }; + /** + * Gets or sets the amount of vertical space in which items are manipulated + * (see {@link legend#spaceY}) + * @param {number} [_=none] should be a number > 0 + * @returns {legend | number} + * @memberof legend + * @property + * by default spaceY = undefined + */ + legend.spaceY = function(_) { return arguments.length ? (spaceY = _, legend) : spaceY; }; + + /** + * Gets / sets whether or not legend is allowed to go beyond specified dimensions + * (see {@link legend#spaceX}) + * @param {boolean} [_=none] + * @returns {legend | boolean} + * @memberof legend + * @property + * by default overflowQ = false + */ + legend.overflowQ = function(_) { return arguments.length ? (overflowQ = _, legend) : overflowQ; }; + /** + * Gets / sets the grouping of the bars + * (see {@link legend#grouping}) + * @param {Array[]} [_=none] + * @returns {legend | Array[]} + * @memberof legend + * @property + * by default grouping = undefined + */ + legend.grouping = function(_) { return arguments.length ? (grouping = _, legend) : grouping; }; + /** + * Gets / sets the valueExtractor + * (see {@link legend#valueExtractor}) + * @param {function} [_=none] + * @returns {legend | function} + * @memberof legend + * @property + * by default valueExtractor = function(key, index) { return data[key] }, + */ + legend.valueExtractor = function(_) { return arguments.length ? (valueExtractor = _, legend) : valueExtractor; }; + /** + * Gets / sets the sortingFunction + * (see {@link bar#sortingFunction}) + * @param {function} [_=none] + * @returns {bar | function} + * @memberof bar + * @property + * by default sortingFunction = function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}, + */ + legend.sortingFunction = function(_) { return arguments.length ? (sortingFunction = _, legend) : sortingFunction; }; + /** + * Gets / sets objectSpacer + * (see {@link legend#objectSpacer}) + * @param {number} [_=none] + * @returns {legend | number} + * @memberof legend + * @property + * by default objectSpacer = 0.05 + */ + legend.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, legend) : objectSpacer; }; + /** + * Gets / sets the minObjectSize + * (see {@link legend#minObjectSize}) + * @param {number} [_=none] + * @returns {legend | number} + * @memberof legend + * @property + * by default minObjectSize = 50 + */ + legend.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, legend) : minObjectSize; }; + /** + * Gets / sets the maxObjectSize + * (see {@link legend#maxObjectSize}) + * @param {number} [_=none] + * @returns {legend | number} + * @memberof legend + * @property + * by default maxObjectSize = 100 + */ + legend.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, legend) : maxObjectSize; }; + + /** + * Gets / sets the barStrokeWidth + * (see {@link legend#barStrokeWidth}) + * @param {number} [_=none] + * @returns {legend | number} + * @memberof legend + * @property + * by default barStrokeWidth = 2 + */ + legend.bubbleStrokeWidth = function(_) { return arguments.length ? (bubbleStrokeWidth = _, legend) : bubbleStrokeWidth; }; + /** + * Gets / sets the colorFunction + * (see {@link legend#colorFunction}) + * @param {number} [_=none] + * @returns {legend | number} + * @memberof legend + * @property + * by default colorFunction = colorFunction() + */ + legend.colorFunction = function(_) { return arguments.length ? (colorFunction$$1 = _, legend) : colorFunction$$1; }; + + /** + * Gets / sets the backgroundFill + * (see {@link legend#backgroundFill}) + * @param {string} [_=none] + * @returns {legend | string} + * @memberof legend + * @property + * by default backgroundFill = 'transparent' + */ + legend.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, legend) : backgroundFill; }; + /** + * Gets / sets the namespace + * (see {@link legend#namespace}) + * @param {string} [_=none] + * @returns {legend | string} + * @memberof legend + * @property + * by default namespace = 'd3sm-legend' + */ + legend.namespace = function(_) { return arguments.length ? (namespace = _, legend) : namespace; }; + /** + * Gets / sets the objectClass + * (see {@link legend#objectClass}) + * @param {string} [_=none] + * @returns {legend | string} + * @memberof legend + * @property + * by default objectClass = 'tick-group' + */ + legend.objectClass = function(_) { return arguments.length ? (objectClass = _, legend) : objectClass; }; + /** + * Gets / sets the transitionDuration + * (see {@link legend#transitionDuration}) + * @param {number} [_=none] + * @returns {legend | number} + * @memberof legend + * @property + * by default transitionDuration = 1000 + */ + legend.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, legend) : transitionDuration; }; + /** + * Gets / sets the easeFunc + * (see {@link legend#easeFunc}) + * @param {d3.ease} [_=none] + * @returns {legend | d3.ease} + * @memberof legend + * @property + * by default easeFunc = d3.easeExp + */ + legend.easeFunc = function(_) { return arguments.length ? (easeFunc = _, legend) : easeFunc; }; + + + function legend() { + var horizontalQ = (orient == 'horizontal') ? true : false; + var verticalQ = !horizontalQ; + // background cliping rectangle + var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; + var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); + + + colorFunction$$1.dataExtent([0, categories.length - 1]) + .colorBy('categories') + .categoryExtractor(function(k, v, i){return v}); + + var r = Math.min(spaceX, spaceY) / 2; + var numberOfObjects = categories.length; + + // if grouping is undefined sort barKeys by sortingFunction + var ordered = (grouping == undefined) ? categories.sort(sortingFunction) : grouping; + // ordered might be nested depending on grouping + var catKeys = utils.arr.flatten(ordered); + + var space = horizontalQ ? spaceX : spaceY; + // calculate object size + var objectSize = utils.math.calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ); + // calculate spacer size if needed + var spacerSize = utils.math.calculateWidthOfSpacer(catKeys, space, objectSize, numberOfObjects, objectSpacer, overflowQ); + // make the nested groups + var spacerFunction = groupingSpacer() + .horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects) + .objectClass(objectClass).objectSize(objectSize).spacerSize(spacerSize) + .transitionDuration(transitionDuration).easeFunc(easeFunc) + .namespace(namespace); + + spacerFunction(container, ordered, 0); + var r = Math.min(objectSize, spaceX, spaceY) / 2 - bubbleStrokeWidth; + + container.selectAll('g:not(.to-remove).'+objectClass).each(function(cat, i) { + var t = d3.select(this); + var c = utils.sel.safeSelect(t, 'circle'); + var fillColor = colorFunction$$1(undefined, cat, i, 'fill'), // prevent duplicate computation + strokeColor = colorFunction$$1(undefined, cat, i, 'stroke'); + + var cx = horizontalQ + ? r+bubbleStrokeWidth + : (spaceX - r*2) / 2 + r; + var cy = verticalQ + ? r+bubbleStrokeWidth + : (spaceX - r*2) / 2 + r; + + c.attr("r", r) + .attr('cx', cx) + .attr('cy', cy) + .attr('fill', fillColor) + .attr('stroke', strokeColor) + .attr('stroke-width', bubbleStrokeWidth); + + var text = utils.sel.safeSelect(t, 'text'); + text.text(cat) + .attr('text-anchor', 'middle') + .attr("transform", function(d, i){ + var + x = cx, + y = cy + text.node().getBoundingClientRect().height / 4, + t = 'translate('+x+','+y+')'; + return t + }); + + }); + + + } + + return legend + } + + function numericLegend( selection ) { + + var + min=0, + max=1, + spaceX, + spaceY, + colorFunction$$1 = colorFunction(), + namespace='d3sm-linear-vertical-gradient', + fontSize = 12, + backgroundFill = 'transparent', + textColor = 'black', + roundTo = 2; + + + legend.min = function(_) { return arguments.length ? (min=_, legend) : min }; + legend.max = function(_) { return arguments.length ? (max=_, legend) : max }; + legend.spaceX = function(_) { return arguments.length ? (spaceX=_, legend) : spaceX }; + legend.spaceY = function(_) { return arguments.length ? (spaceY=_, legend) : spaceY }; + legend.namespace = function(_) { return arguments.length ? (namespace=_, legend) : namespace }; + legend.fontSize = function(_) { return arguments.length ? (fontSize=_, legend) : fontSize }; + legend.backgroundFill = function(_) { return arguments.length ? (backgroundFill=_, legend) : backgroundFill }; + legend.colorFunction = function(_) { return arguments.length ? (colorFunction$$1=_, legend) : colorFunction$$1 }; + legend.textColor = function(_) { return arguments.length ? (textColor=_, legend) : textColor }; + legend.roundTo = function(_) { return arguments.length ? (roundTo=_, legend) : roundTo }; + + function legend() { + // background cliping rectangle + var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; + var container = utils.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); + + var defs = utils.sel.safeSelect(selection, 'defs'); + var linearGradient = utils.sel.safeSelect(defs, 'linearGradient') + .attr("x1", "0%") + .attr("y1", "100%") + .attr("x2", "0%") + .attr("y2", "0%") + .attr('id', utils.str.hypenate(namespace,'numerical-legend-gradient')); + + + colorFunction$$1.dataExtent([min, max]) + .colorBy('value') + .valueExtractor(function(k, v, i){return v}); + + + linearGradient.selectAll('stop') + .data( colorFunction$$1.colors() ) + .enter() + .append('stop') + .attr("offset", function(d, i){ return i / (colorFunction$$1.colors().length - 1) }) + .attr('stop-color', function(d) {return d}); + + + + + var rect = utils.sel.safeSelect(container, 'rect', 'legend') + .attr('transform', 'translate(0,'+fontSize+')') + .style("fill", "url(#"+utils.str.hypenate(namespace,'numerical-legend-gradient')+")") + .attr('x', 0) + .attr('y', 0) + .attr('width', spaceX) + .attr('height', spaceY - fontSize*2) + .on('mousemove', function(d, i){legendMousemove(d, i, rect, d3.select(this));}) + .on('mouseout', function(d, i){ d3.select("#"+utils.str.hypenate(namespace,'legend-tooltip')).remove(); }); + + var minText = utils.sel.safeSelect(container, 'text', 'min') + .text(utils.math.round(min, 2)) + .attr('text-anchor', 'middle') + .attr("font-size", fontSize+'px') + .attr('transform', function(d, i){ + var + x = spaceX / 2, + y = spaceY - fontSize / 4, + t = 'translate('+x+','+y+')'; + return t + }); + + var maxText = utils.sel.safeSelect(container, 'text', 'max') + .text(utils.math.round(max, 2)) + .attr('text-anchor', 'middle') + .attr("font-size", fontSize+'px') + .attr('transform', function(d, i){ + var + x = spaceX / 2, + y = fontSize, + t = 'translate('+x+','+y+')'; + return t + }); + + + + + } + + function legendMousemove(d, i, rect, t) { + var s = d3.scaleLinear() + .domain([0, rect.attr('height')]) + .range([max, min]); + var m = d3.mouse(rect.node()); + var v = utils.math.round(s(m[1]),roundTo); + + var strokeColor = colorFunction$$1(undefined, v, undefined, 'stroke'); + var fillColor = colorFunction$$1(undefined, v, undefined, 'fill'); + + var div = utils.sel.safeSelect(d3.select('body'), 'div', utils.str.hypenate(namespace,'legend-tooltip')) + .attr('id', utils.str.hypenate(namespace,'legend-tooltip')) + .style('position', 'absolute') + .style('left', (d3.event.pageX+15)+'px') + .style('top', (d3.event.pageY+15)+'px') + .style('background-color', fillColor) + .style('border-color', strokeColor) + + .style('min-width', (fontSize * (String(max).split('.')[0].length+3))+'px') + .style('min-height', (fontSize * (String(max).split('.')[0].length+3))+'px') + .style('border-radius', '50%') + .style('border-radius', '5000px') + + .style('display', 'flex') + .style('justify-content', 'center') + .style('text-align', 'middle') + .style('padding', 2+"px") + + .style('border-style', 'solid') + .style('border-width', 2); + + var text = utils.sel.safeSelect(div, 'div') + .text(v) + .style('color', textColor) + .style('align-self', 'center'); + } + + return legend + } + + let legends = { + categorical: categoricLegend, numeric: numericLegend + }; + + // import * as d3 from "d3"; + /******************************************************************************* + ** ** + ** ** + ** D3 EXTENSIONS ** + ** ** + ** ** + *******************************************************************************/ + /** + * Recursively ascends parents of selection until it finds an svg tag + * @function d3.selection.thisSVG + * @augments d3.selection + * @returns {Element} which is the svg tag, not the d3 selection of that tag + */ + d3.selection.prototype.thisSVG = function() { return utils.sel.getContainingSVG(this.node()); }; + + + /** + * Helper for getting absolute position of the mouse + * @function d3.mouse.absolute + * @augments d3.mouse + * @returns {number[]} [x, y] as they relate to `html` not to local scope. + */ + d3.mouse.absolute = function() { + var html = d3.select('html').node(); + var [x, y] = this(html); + return [x, y] + }; + + + /** + * Gets position of the selection in relation to the containing svg + * @see{@link getContainingSVG} + * @function d3.selection.absolutePosition + * @augments d3.selection + * @returns {Object} with structure similar to getBoundingClientRect, e.g. + * top, left, bottom, right, height, width + */ + d3.selection.prototype.absolutePosition = function() { + var element = this.node(); + var elementPosition = element.getBoundingClientRect(); + var containerSVG = utils.sel.getContainingSVG(element); + var svgPosition = containerSVG.getBoundingClientRect(); + + return { + top: elementPosition.top - svgPosition.top, + left: elementPosition.left - svgPosition.left, + bottom: elementPosition.bottom - svgPosition.top, + right: elementPosition.right - svgPosition.left, + height: elementPosition.height, + width: elementPosition.width + }; + + }; + + + d3.selection.prototype.relativePositionTo = function(container) { + var element = this.node(); + var elementPosition = element.getBoundingClientRect(); + var containerSVG = container; + var svgPosition = containerSVG.getBoundingClientRect(); + + return { + top: elementPosition.top - svgPosition.top, + left: elementPosition.left - svgPosition.left, + bottom: elementPosition.bottom - svgPosition.top, + right: elementPosition.right - svgPosition.left, + height: elementPosition.height, + width: elementPosition.width + }; + + }; + + function getTranslation$1(selection){ + var transform = selection.attr('transform'); + var [junk, xy] =transform.split('translate('); + var [x, y] = xy.split(','); + junk = y.split(')'); + return [parseFloat(x), parseFloat(y)] + } + + function lasso( selection ) { + var + svg, // svg that is target of events + objectContainer, // container which houses objects we are selecting (allows for transform to be applied to lasso) + objectClass, // class of object we are selecting + namespace="d3sm-lasso", + chartContainer, + chartOffset, + objectsOffset, + eventCatcher, + + xScale, // optional scale for the lasso currentPoints + yScale, // optional scale for the lasso currentPoints + + activeQ = false, // whether or not lasso is active + + currentPoints=[], // mouse points for current lasso + allPoints=[], // list of lists for all points of lassos + + line = d3.line() + .x(function(d, i){ + var x; + if (xScale != undefined) { x = xScale(d[0]); } + else {x = d[0];} + return x //- chartOffset[0]// - objectsOffset[0] + }) + .y(function(d, i){ + var y; + if (yScale != undefined) { y= yScale(d[1]); } + else {y = d[1];} + return y// - chartOffset[1]// - objectsOffset[1] + }) + .curve(d3.curveLinearClosed), + + instance=0, // an indentifier for which instance this lasso is under the current svg + + tickDistance = 10, + + // styles for lasso path + color$$1 = '#17a2b8', + animationRate = '10s', + opacity=0.3, + dashArray = '5, 10', + stroke = 'black', + strokeWidth=2, + + // styles for lassoed objects + lassoedFill = "white", + lassoedStroke = 'black', + lassoedStrokeWidth = 3, + + transitionDuration = 1000, + easeFunc = d3.easeExp; + + var path; + + lasso.svg = function(_) { return arguments.length ? (svg = _, lasso) : svg; }; + lasso.chartContainer = function(_) { return arguments.length ? (chartContainer = _, lasso) : chartContainer; }; + lasso.objectContainer = function(_) { return arguments.length ? (objectContainer = _, lasso) : objectContainer; }; + lasso.objectClass = function(_) { return arguments.length ? (objectClass = _, lasso) : objectClass; }; + lasso.namespace = function(_) { return arguments.length ? (namespace = _, lasso) : namespace; }; + lasso.xScale = function(_) { return arguments.length ? (xScale = _, lasso) : xScale; }; + lasso.yScale = function(_) { return arguments.length ? (yScale = _, lasso) : yScale; }; + lasso.activeQ = function(_) { return arguments.length ? (activeQ = _, lasso) : activeQ; }; + lasso.currentPoints = function(_) { return arguments.length ? (currentPoints = _, lasso) : currentPoints; }; + lasso.allPoints = function(_) { return arguments.length ? (allPoints = _, lasso) : allPoints; }; + lasso.instance = function(_) { return arguments.length ? (instance = _, lasso) : instance; }; + lasso.tickDistance = function(_) { return arguments.length ? (tickDistance = _, lasso) : tickDistance; }; + lasso.color = function(_) { return arguments.length ? (color$$1 = _, lasso) : color$$1; }; + lasso.animationRate = function(_) { return arguments.length ? (animationRate = _, lasso) : animationRate; }; + lasso.opacity = function(_) { return arguments.length ? (opacity = _, lasso) : opacity; }; + lasso.dashArray = function(_) { return arguments.length ? (dashArray = _, lasso) : dashArray; }; + lasso.stroke = function(_) { return arguments.length ? (stroke = _, lasso) : stroke; }; + lasso.lassoedFill = function(_) { return arguments.length ? (lassoedFill = _, lasso) : lassoedFill; }; + lasso.lassoedStroke = function(_) { return arguments.length ? (lassoedStroke = _, lasso) : lassoedStroke; }; + lasso.lassoedStrokeWidth = function(_) { return arguments.length ? (lassoedStrokeWidth = _, lasso) : lassoedStrokeWidth; }; + lasso.eventCatcher = function(_) { return arguments.length ? (eventCatcher = _, lasso) : eventCatcher; }; + + lasso.drag = drag; + lasso.draw = draw; + lasso.tick = tick; + lasso.detect = detect; + lasso.toggle = toggle; + lasso.remove = remove; + lasso.render = render; + lasso.keyFrames = keyFrames; + lasso.updateObjects = updateObjects; + lasso.applyPathAttributes = applyPathAttributes; + lasso.applyObjectAttributes = applyObjectAttributes; + + keyFrames(); + + function lasso() { + // add a dash animation if needed + if (activeQ) { transitionDraw(); } + } + + function toggle(state) { + // use optional param to set state, otherwise toggle state + activeQ = (state!=undefined) ? state : !activeQ; + chartOffset = getTranslation$1(chartContainer); //utils.sel.getTranslation(chartContainer) + objectsOffset = getTranslation$1(objectContainer); //utils.sel.getTranslation(objectContainer) + + if (activeQ) { + svg.node().addEventListener('mousedown', render, true); + } else { + svg.node().removeEventListener('mousedown', render, true); + remove(); + } + + } + + function draw() { + chartOffset = getTranslation$1(chartContainer); //utils.sel.getTranslation(chartContainer) + objectsOffset = getTranslation$1(objectContainer); //utils.sel.getTranslation(objectContainer) + + var container = utils.sel.safeSelect(objectContainer, 'g', 'lasso-container'); + var paths$$1 = container.selectAll('path[instance="'+instance+'"]'); + + // update + paths$$1 = paths$$1.data(allPoints); + + // remove excess + var pExit = paths$$1.exit().remove(); + // add needed paths + var pEnter = paths$$1.enter().append('path'); + + // merge + paths$$1 = paths$$1.merge(pEnter); + + // apply + applyPathAttributes(paths$$1); + } + + function remove() { + var container = utils.sel.safeSelect(objectContainer, 'g', 'lasso-container'); + var paths$$1 = container.selectAll('path[instance="'+instance+'"]').remove(); + container.remove(); + objectContainer.selectAll(objectClass).classed("in-lasso", false); + updateObjects(); + } + + function render( event ) { + // nothing can interefer with drawing the lasso + event.preventDefault(); event.stopPropagation(); + + var container = utils.sel.safeSelect(objectContainer, 'g', 'lasso-container'); + + /* + each time the user presses down, while the state is active, the lasso + the lasso should make a seperate segment. + */ + currentPoints = []; + + svg.node().addEventListener('mousemove', drag); + svg.node().addEventListener('mouseup', function(event) { + svg.node().removeEventListener('mousemove', drag); + allPoints.push(currentPoints); + // BUG: somehow this is pushing currentPoints n times where n is the nth lasso path for the current instance + // NOTE: allPoints = utils.arr.unique(allPoints) is a temporary and inefficient fix + allPoints = utils.arr.unique(allPoints); + }); + + path = container.append('path').data([currentPoints]); + applyPathAttributes(path); + } + + function transitionDraw() { + var container = utils.sel.safeSelect(objectContainer, 'g', 'lasso-container'); + var paths$$1 = container.selectAll('path[instance="'+instance+'"]'); + + // update + paths$$1 = paths$$1.data(allPoints); + + // remove excess + var pExit = paths$$1.exit().remove(); + // add needed paths + var pEnter = paths$$1.enter().append('path'); + + // merge + paths$$1 = paths$$1.merge(pEnter) + .transition().duration(transitionDuration) + .ease(easeFunc); + applyPathAttributes(paths$$1); + + } + + function applyPathAttributes(path) { + path + .attr("class", utils.str.hypenate(namespace, "lasso-path")) + .style('opacity', opacity) + .attr('fill', color$$1) + .attr("d", line) + .attr('instance', instance) + .style("stroke-dasharray", dashArray) + .attr("stroke", stroke) + .attr("stroke-width", strokeWidth) + .style('animation', 'lassoDash '+animationRate+' linear') + .style("animation-iteration-count", "infinite"); + } - /** - * Duration of all transitions of this element - * @param {number} [transitionDuration=1000] - * @memberof legend# - * @property - */ - transitionDuration = 1000, - /** - * Easing function for transitions - * @param {d3.ease} [easeFunc=d3.easeExp] - * @memberof legend# - * @property - */ - easeFunc = d3.easeExp; + function drag(event) { + /* + effectively create a mouse down and move event (which normally is inteperated + as 'drag' by the browser) by dynamically adding / removing this event on + mouse down / mouse up. + */ + if (eventCatcher != undefined) {eventCatcher.dispatch(utils.str.hypenate(namespace,"drag"));} + // d3.dispatch(utils.str.hypenate(namespace,"drag")) - legend.categories = function(_) { return arguments.length ? (categories=_, legend) : categories }; + if (event.which != 1) {return} // ensures left mouse button set + d3.event = event; + var pt = d3.mouse(objectContainer.node()); + var pt = d3.mouse(svg.node()); - /** - * Gets or sets the selection in which items are manipulated - * @param {d3.selection} [_=none] - * @returns {legend | d3.selection} - * @memberof legend - * @property - * by default selection = selection - */ - legend.selection = function(_) { return arguments.length ? (selection = _, legend) : selection; }; - /** - * Gets or sets the data - * (see {@link legend#data}) - * @param {number} [_=none] - * @returns {legend | object} - * @memberof legend - * @property - */ - legend.data = function(_) { return arguments.length ? (data = _, legend) : data; }; - /** - * Gets or sets the orient of the bars - * (see {@link legend#orient}) - * @param {number} [_=none] - * @returns {legend | object} - * @memberof legend - * @property - */ - legend.orient = function(_) { return arguments.length ? (orient = _, legend) : orient; }; - /** - * Gets or sets the amount of horizontal space in which items are manipulated - * (see {@link legend#spaceX}) - * @param {number} [_=none] should be a number > 0 - * @returns {legend | number} - * @memberof legend - * @property - * by default spaceX = undefined - */ - legend.spaceX = function(_) { return arguments.length ? (spaceX = _, legend) : spaceX; }; - /** - * Gets or sets the amount of vertical space in which items are manipulated - * (see {@link legend#spaceY}) - * @param {number} [_=none] should be a number > 0 - * @returns {legend | number} - * @memberof legend - * @property - * by default spaceY = undefined - */ - legend.spaceY = function(_) { return arguments.length ? (spaceY = _, legend) : spaceY; }; + if (xScale != undefined) {pt[0] = xScale.invert(pt[0]);} + if (yScale != undefined) {pt[1] = yScale.invert(pt[1]);} + pt[0] = pt[0] - chartOffset[0] - objectsOffset[0]; + pt[1] = pt[1] - chartOffset[1] - objectsOffset[1]; - /** - * Gets / sets whether or not legend is allowed to go beyond specified dimensions - * (see {@link legend#spaceX}) - * @param {boolean} [_=none] - * @returns {legend | boolean} - * @memberof legend - * @property - * by default overflowQ = false - */ - legend.overflowQ = function(_) { return arguments.length ? (overflowQ = _, legend) : overflowQ; }; - /** - * Gets / sets the grouping of the bars - * (see {@link legend#grouping}) - * @param {Array[]} [_=none] - * @returns {legend | Array[]} - * @memberof legend - * @property - * by default grouping = undefined - */ - legend.grouping = function(_) { return arguments.length ? (grouping = _, legend) : grouping; }; - /** - * Gets / sets the valueExtractor - * (see {@link legend#valueExtractor}) - * @param {function} [_=none] - * @returns {legend | function} - * @memberof legend - * @property - * by default valueExtractor = function(key, index) { return data[key] }, - */ - legend.valueExtractor = function(_) { return arguments.length ? (valueExtractor = _, legend) : valueExtractor; }; - /** - * Gets / sets the sortingFunction - * (see {@link bar#sortingFunction}) - * @param {function} [_=none] - * @returns {bar | function} - * @memberof bar - * @property - * by default sortingFunction = function(keyA, keyB) {return d3.descending(data[keyA], data[keyB])}, - */ - legend.sortingFunction = function(_) { return arguments.length ? (sortingFunction = _, legend) : sortingFunction; }; - /** - * Gets / sets objectSpacer - * (see {@link legend#objectSpacer}) - * @param {number} [_=none] - * @returns {legend | number} - * @memberof legend - * @property - * by default objectSpacer = 0.05 - */ - legend.objectSpacer = function(_) { return arguments.length ? (objectSpacer = _, legend) : objectSpacer; }; - /** - * Gets / sets the minObjectSize - * (see {@link legend#minObjectSize}) - * @param {number} [_=none] - * @returns {legend | number} - * @memberof legend - * @property - * by default minObjectSize = 50 - */ - legend.minObjectSize = function(_) { return arguments.length ? (minObjectSize = _, legend) : minObjectSize; }; - /** - * Gets / sets the maxObjectSize - * (see {@link legend#maxObjectSize}) - * @param {number} [_=none] - * @returns {legend | number} - * @memberof legend - * @property - * by default maxObjectSize = 100 - */ - legend.maxObjectSize = function(_) { return arguments.length ? (maxObjectSize = _, legend) : maxObjectSize; }; + /* if we have a point already, test if it passes a minimum distance to prevent overwhelming with too many tick functions */ + if (currentPoints.length) { + var lastPt = currentPoints[currentPoints.length - 1]; + var a = [pt[0], pt[1]], b = [lastPt[0], lastPt[1]]; - /** - * Gets / sets the barStrokeWidth - * (see {@link legend#barStrokeWidth}) - * @param {number} [_=none] - * @returns {legend | number} - * @memberof legend - * @property - * by default barStrokeWidth = 2 - */ - legend.bubbleStrokeWidth = function(_) { return arguments.length ? (bubbleStrokeWidth = _, legend) : bubbleStrokeWidth; }; - /** - * Gets / sets the colorFunction - * (see {@link legend#colorFunction}) - * @param {number} [_=none] - * @returns {legend | number} - * @memberof legend - * @property - * by default colorFunction = colorFunction() - */ - legend.colorFunction = function(_) { return arguments.length ? (colorFunction$$1 = _, legend) : colorFunction$$1; }; + if (xScale) {b[0] = xScale(b[0]); a[0] = xScale(a[0]);} + if (yScale) {b[1] = yScale(b[1]); a[1] = yScale(a[1]);} - /** - * Gets / sets the backgroundFill - * (see {@link legend#backgroundFill}) - * @param {string} [_=none] - * @returns {legend | string} - * @memberof legend - * @property - * by default backgroundFill = 'transparent' - */ - legend.backgroundFill = function(_) { return arguments.length ? (backgroundFill = _, legend) : backgroundFill; }; - /** - * Gets / sets the namespace - * (see {@link legend#namespace}) - * @param {string} [_=none] - * @returns {legend | string} - * @memberof legend - * @property - * by default namespace = 'd3sm-legend' - */ - legend.namespace = function(_) { return arguments.length ? (namespace = _, legend) : namespace; }; - /** - * Gets / sets the objectClass - * (see {@link legend#objectClass}) - * @param {string} [_=none] - * @returns {legend | string} - * @memberof legend - * @property - * by default objectClass = 'tick-group' - */ - legend.objectClass = function(_) { return arguments.length ? (objectClass = _, legend) : objectClass; }; - /** - * Gets / sets the transitionDuration - * (see {@link legend#transitionDuration}) - * @param {number} [_=none] - * @returns {legend | number} - * @memberof legend - * @property - * by default transitionDuration = 1000 - */ - legend.transitionDuration = function(_) { return arguments.length ? (transitionDuration = _, legend) : transitionDuration; }; - /** - * Gets / sets the easeFunc - * (see {@link legend#easeFunc}) - * @param {d3.ease} [_=none] - * @returns {legend | d3.ease} - * @memberof legend - * @property - * by default easeFunc = d3.easeExp - */ - legend.easeFunc = function(_) { return arguments.length ? (easeFunc = _, legend) : easeFunc; }; + var dist = utils.math.euclideanDistance(b, a); + if (dist > tickDistance) { tick(pt); } + } + else { tick(pt); } + } - function legend() { - var horizontalQ = (orient == 'horizontal') ? true : false; - var verticalQ = !horizontalQ; - // background cliping rectangle - var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; - var container = utils$1.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); + function tick (pt) { + /* + If a point is provided update data and objects. + Otherwise just call on data we already have. + Why like this?: + 1. currentPoints is current points to allow disjunct lassos, currentPoints is only pushed to + allPoints after mouseup. + 2. to allow render of objects in the lasso class / updating the data list + just by toggling the button + */ - colorFunction$$1.dataExtent([0, categories.length - 1]) - .colorBy('categories') - .categoryExtractor(function(k, v, i){return v}); + if (pt != undefined) { + currentPoints.push(pt); + path.attr("d", line); + if (currentPoints.length < 3) {return} // need at least 3 points to detect anything. + detect(allPoints.concat([currentPoints])); + } else { + detect(allPoints); + } + } - var r = Math.min(spaceX, spaceY) / 2; - var numberOfObjects = categories.length; - // if grouping is undefined sort barKeys by sortingFunction - var ordered = (grouping == undefined) ? categories.sort(sortingFunction) : grouping; - // ordered might be nested depending on grouping - var catKeys = utils$1.arr.flatten(ordered); + function detect(lassos) { + if (lassos == undefined) {lassos = allPoints;} + objectContainer.selectAll(objectClass).each(function(d, i){ + var current = d3.select(this), - var space = horizontalQ ? spaceX : spaceY; - // calculate object size - var objectSize = utils$1.math.calculateWidthOfObject(space, numberOfObjects, minObjectSize, maxObjectSize, objectSpacer, overflowQ); - // calculate spacer size if needed - var spacerSize = utils$1.math.calculateWidthOfSpacer(catKeys, space, objectSize, numberOfObjects, objectSpacer, overflowQ); - // make the nested groups - var spacerFunction = groupingSpacer() - .horizontalQ(horizontalQ).scale(scale).moveby('category').numberOfObjects(numberOfObjects) - .objectClass(objectClass).objectSize(objectSize).spacerSize(spacerSize) - .transitionDuration(transitionDuration).easeFunc(easeFunc) - .namespace(namespace); + box = current.absolutePosition(), + // box = current.relativePositionTo(objectContainer.node()), - spacerFunction(container, ordered, 0); - var r = Math.min(objectSize, spaceX, spaceY) / 2 - bubbleStrokeWidth; + boxPts = [ + [ + box.left - chartOffset[0] - objectsOffset[0], + box.top - chartOffset[1] - objectsOffset[1] + ], + [ + box.right - chartOffset[0] - objectsOffset[0], + box.top - chartOffset[1] - objectsOffset[1] + ], + [ + box.left - chartOffset[0] - objectsOffset[0], + box.bottom - chartOffset[1] - objectsOffset[1] + ], + [ + box.right - chartOffset[0] - objectsOffset[0], + box.bottom - chartOffset[1] - objectsOffset[1] + ] + ]; - container.selectAll('g:not(.to-remove).'+objectClass).each(function(cat, i) { - var t = d3.select(this); - var c = utils$1.sel.safeSelect(t, 'circle'); - var fillColor = colorFunction$$1(undefined, cat, i, 'fill'), // prevent duplicate computation - strokeColor = colorFunction$$1(undefined, cat, i, 'stroke'); + if (xScale != undefined) { + boxPts[0][0] = xScale.invert(boxPts[0][0]); + boxPts[1][0] = xScale.invert(boxPts[1][0]); + boxPts[2][0] = xScale.invert(boxPts[2][0]); + boxPts[3][0] = xScale.invert(boxPts[3][0]); + } + if (yScale != undefined) { + boxPts[0][1] = yScale.invert(boxPts[0][1]); + boxPts[1][1] = yScale.invert(boxPts[1][1]); + boxPts[2][1] = yScale.invert(boxPts[2][1]); + boxPts[3][1] = yScale.invert(boxPts[3][1]); + } - var cx = horizontalQ - ? r+bubbleStrokeWidth - : (spaceX - r*2) / 2 + r; - var cy = verticalQ - ? r+bubbleStrokeWidth - : (spaceX - r*2) / 2 + r; - c.attr("r", r) - .attr('cx', cx) - .attr('cy', cy) - .attr('fill', fillColor) - .attr('stroke', strokeColor) - .attr('stroke-width', bubbleStrokeWidth); + /* + flag needed as we have to test multiple lasso segments, and if the point + is not in one segment, it does not mean it is not in any + */ + var inAnyLassoQ = false; + for (var i = 0; i < lassos.length; i++) { + var lassoPoints = lassos[i]; + // .map(function(pt){ + // var x, y = pt + // if (xScale!=undefined) {x = xScale(x)} + // if (yScale!=undefined) {y = yScale(y)} + // return [x, y] + // }) + var boxInLassoQ = boxPts.every(coord => d3.polygonContains(lassoPoints, coord)); - var text = utils$1.sel.safeSelect(t, 'text'); - text.text(cat) - .attr('text-anchor', 'middle') - .attr("transform", function(d, i){ - var - x = cx, - y = cy + text.node().getBoundingClientRect().height / 4, - t = 'translate('+x+','+y+')'; - return t - }); + if (boxInLassoQ) { inAnyLassoQ = true; } // only update flag in the positive case. + } + current.classed('in-lasso', inAnyLassoQ); + current.classed('in-lasso-'+instance, inAnyLassoQ); }); - + updateObjects(); + return objectContainer.selectAll('.in-lasso-'+instance) } - return legend - } - function numericLegend( selection ) { - var - min=0, - max=1, - spaceX, - spaceY, - colorFunction$$1 = colorFunction(), - namespace='d3sm-linear-vertical-gradient', - fontSize = 12, - backgroundFill = 'transparent', - textColor = 'black', - roundTo = 2; + function updateObjects() { + objectContainer.selectAll(objectClass).each(function(d, i) { + var t = d3.select(this); + applyObjectAttributes(t, t.classed('in-lasso')); + }); + } + function applyObjectAttributes(obj, setQ) { + var + preLassoFill = obj.attr('_pre_lasso_fill'), + preLassoStroke = obj.attr('_pre_lasso_stroke'), + preLassoStrokeWidth = obj.attr('_pre_lasso_stroke-width'); - legend.min = function(_) { return arguments.length ? (min=_, legend) : min }; - legend.max = function(_) { return arguments.length ? (max=_, legend) : max }; - legend.spaceX = function(_) { return arguments.length ? (spaceX=_, legend) : spaceX }; - legend.spaceY = function(_) { return arguments.length ? (spaceY=_, legend) : spaceY }; - legend.namespace = function(_) { return arguments.length ? (namespace=_, legend) : namespace }; - legend.fontSize = function(_) { return arguments.length ? (fontSize=_, legend) : fontSize }; - legend.backgroundFill = function(_) { return arguments.length ? (backgroundFill=_, legend) : backgroundFill }; - legend.colorFunction = function(_) { return arguments.length ? (colorFunction$$1=_, legend) : colorFunction$$1 }; - legend.textColor = function(_) { return arguments.length ? (textColor=_, legend) : textColor }; - legend.roundTo = function(_) { return arguments.length ? (roundTo=_, legend) : roundTo }; + if (setQ) { + obj.classed("in-lasso", true); + obj.classed('in-lasso-'+instance, true); + if (preLassoFill == undefined) { obj.attr('_pre_lasso_fill', obj.attr('fill')); } + if (preLassoStroke == undefined) { obj.attr('_pre_lasso_stroke', obj.attr('stroke')); } + if (preLassoStrokeWidth == undefined) { obj.attr('_pre_lasso_stroke-width', obj.attr('stroke-width')); } - function legend() { - // background cliping rectangle - var bgcpRect = {x:0, y:0, width: spaceX, height:spaceY}; - var container = utils$1.sel.setupContainer( selection, namespace, bgcpRect, backgroundFill ); + obj + //BUG: when .raise() + .attr('fill', lassoedFill) + .attr('stroke', lassoedStroke) + .attr('stoke-width', lassoedStrokeWidth); - var defs = utils$1.sel.safeSelect(selection, 'defs'); - var linearGradient = utils$1.sel.safeSelect(defs, 'linearGradient') - .attr("x1", "0%") - .attr("y1", "100%") - .attr("x2", "0%") - .attr("y2", "0%") - .attr('id', utils$1.str.hypenate(namespace,'numerical-legend-gradient')); + } else { + obj.classed("in-lasso", false); + obj.classed('in-lasso-'+instance, false); + if (preLassoFill != undefined) { obj.attr('fill', preLassoFill); } + if (preLassoStroke != undefined) { obj.attr('stroke', preLassoStroke); } + if (preLassoStrokeWidth != undefined) { obj.attr('stroke-width', preLassoStrokeWidth); } + } + } + function keyFrames() { + var style = + d3.select("html").select('style.'+utils.str.hypenate(namespace,"lasso-dash")); + if (style.empty()) { + d3.select("html").append('style') + .classed(utils.str.hypenate(namespace,"lasso-dash"), true) + .html("@keyframes lassoDash {to { stroke-dashoffset: 1000;}}"); + } - colorFunction$$1.dataExtent([min, max]) - .colorBy('value') - .valueExtractor(function(k, v, i){return v}); + } + return lasso + } + function lassoWidget( selection ) { + var + namespace = 'd3sm-lasso', + selection = selection, + svg, + chartContainer, + objectContainer, + objectClass, + lassoLine = d3.line() + .x(function(d, i){return d[0]}) + .y(function(d, i){return d[1]}) + .curve(d3.curveLinearClosed), + colorFunction$$1 = colorFunction(), + maxNumberOfGroups, + dataExtractor, + xScale, + yScale; - linearGradient.selectAll('stop') - .data( colorFunction$$1.colors() ) - .enter() - .append('stop') - .attr("offset", function(d, i){ return i / (colorFunction$$1.colors().length - 1) }) - .attr('stop-color', function(d) {return d}); + function onError(msg, style){ + console.log(msg, style); + } + // setup the container + setupDataselectContainer(); + + lassoWidget.objectClass = function(_){return arguments.length ? (objectClass='.'+_.replace('.',''), lassoWidget) : objectClass}; + lassoWidget.svg = function(_){return arguments.length ? (svg=_, lassoWidget) : svg}; + lassoWidget.submit = function(_){return arguments.length ? (submit=_, lassoWidget) : submit}; + lassoWidget.maxNumberOfGroups = function(_){return arguments.length ? (maxNumberOfGroups=_, lassoWidget) : maxNumberOfGroups}; + lassoWidget.onError = function(_){return arguments.length ? (onError=_, lassoWidget) : onError}; + lassoWidget.objectContainer = function(_){return arguments.length ? (objectContainer=_, lassoWidget) : objectContainer}; + lassoWidget.chartContainer = function(_){return arguments.length ? (chartContainer=_, lassoWidget) : chartContainer}; + lassoWidget.dataExtractor = function(_){return arguments.length ? (dataExtractor=_, lassoWidget) : dataExtractor}; + lassoWidget.xScale = function(_){return arguments.length ? (xScale=_, lassoWidget) : xScale}; + lassoWidget.yScale = function(_){return arguments.length ? (yScale=_, lassoWidget) : yScale}; + + function styles() { + var s = d3.select("html").select("style."+namespace+'lasso-widget'); + if (s.empty()) { + d3.select("html").append("style") + .classed(namespace+'lasso-widget', true) + .html( + "."+utils.str.hypenate(namespace, "data-table") + "{\ + height:100px;\ + overflow:auto;\ + }" + ); + } + } + function lassoWidget() { + styles(); + } - var rect = utils$1.sel.safeSelect(container, 'rect', 'legend') - .attr('transform', 'translate(0,'+fontSize+')') - .style("fill", "url(#"+utils$1.str.hypenate(namespace,'numerical-legend-gradient')+")") - .attr('x', 0) - .attr('y', 0) - .attr('width', spaceX) - .attr('height', spaceY - fontSize*2) - .on('mousemove', function(d, i){legendMousemove(d, i, rect, d3.select(this));}) - .on('mouseout', function(d, i){ d3.select("#"+utils$1.str.hypenate(namespace,'legend-tooltip')).remove(); }); + function submit(data) { + console.log(data); + } - var minText = utils$1.sel.safeSelect(container, 'text', 'min') - .text(utils$1.math.round(min, 2)) - .attr('text-anchor', 'middle') - .attr("font-size", fontSize+'px') - .attr('transform', function(d, i){ - var - x = spaceX / 2, - y = spaceY - fontSize / 4, - t = 'translate('+x+','+y+')'; - return t - }); + function plusTab(tabList) { + var tab = utils.sel.safeSelect(tabList, 'li', utils.str.hypenate(namespace, 'plus-tab')) + .classed('ml-auto', true) + .classed('nav-item', true) + .on('click', makeNewGroup), - var maxText = utils$1.sel.safeSelect(container, 'text', 'max') - .text(utils$1.math.round(max, 2)) - .attr('text-anchor', 'middle') - .attr("font-size", fontSize+'px') - .attr('transform', function(d, i){ - var - x = spaceX / 2, - y = fontSize, - t = 'translate('+x+','+y+')'; - return t - }); + anchor = utils.sel.safeSelect(tab, 'a', 'nav-link'), + icon = utils.sel.safeSelect(anchor, 'i', 'fa') + .classed('fa-plus fa-2x text-success', true); + } + + function sendTab(tabList) { + var tab = utils.sel.safeSelect(tabList, 'li', utils.str.hypenate(namespace, 'send-tab')) + .classed('ml-auto', true) + .classed('nav-item', true) + .on('click', clickSend) + , + anchor = utils.sel.safeSelect(tab, 'a', 'nav-link'), + icon = utils.sel.safeSelect(anchor, 'i', 'fa') + .classed('fa-paper-plane-o fa-2x text-primary', true); + } + function clickSend() { + var data = gatherDataLists(); + submit(data); + } + function closeTab(tabList) { + var tab = utils.sel.safeSelect(tabList, 'li', utils.str.hypenate(namespace, 'close-tab')) + .classed('ml-auto', true) + .classed('nav-item', true) + .on('click', function(){ + var m = d3.mouse(d3.select("html").node()); + selection.select('div.card').style("position", 'relative') + .transition().duration(1000) + .ease(d3.easeBack) + .style('left', window.outerWidth +'px') + .remove(); + }) + , + anchor = utils.sel.safeSelect(tab, 'a', 'nav-link'), + icon = utils.sel.safeSelect(anchor, 'i', 'fa') + .classed('fa-window-close-o fa-2x text-danger', true); } - function legendMousemove(d, i, rect, t) { - var s = d3.scaleLinear() - .domain([0, rect.attr('height')]) - .range([max, min]); - var m = d3.mouse(rect.node()); - var v = utils$1.math.round(s(m[1]),roundTo); - var strokeColor = colorFunction$$1(undefined, v, undefined, 'stroke'); - var fillColor = colorFunction$$1(undefined, v, undefined, 'fill'); - var div = utils$1.sel.safeSelect(d3.select('body'), 'div', utils$1.str.hypenate(namespace,'legend-tooltip')) - .attr('id', utils$1.str.hypenate(namespace,'legend-tooltip')) - .style('position', 'absolute') - .style('left', (d3.event.pageX+15)+'px') - .style('top', (d3.event.pageY+15)+'px') - .style('background-color', fillColor) - .style('border-color', strokeColor) + function setupDataselectContainer() { + var + card = utils.sel.safeSelect(selection, 'div', 'card'), + cardHeader = utils.sel.safeSelect(card, 'div', 'card-header'), - .style('min-width', (fontSize * (String(max).split('.')[0].length+3))+'px') - .style('min-height', (fontSize * (String(max).split('.')[0].length+3))+'px') - .style('border-radius', '50%') - .style('border-radius', '5000px') + tabList = utils.sel.safeSelect(cardHeader, 'ul', 'nav') + .classed('nav-tabs card-header-tabs', true) + .classed(utils.str.hypenate(namespace, 'tab-list'), true) + .attr('role', 'tablist'), - .style('display', 'flex') - .style('justify-content', 'center') - .style('text-align', 'middle') - .style('padding', 2+"px") + cardBody = utils.sel.safeSelect(card, 'div', 'card-body'), + tabContent = utils.sel.safeSelect(cardBody, 'div', 'tab-content') + .classed(utils.str.hypenate(namespace, 'tab-content'), true), - .style('border-style', 'solid') - .style('border-width', 2); + // leftTabs = utils.sel.safeSelect(tabList, 'div', 'nav mr-auto left-aligned-tabs') + rightTabs = utils.sel.safeSelect(tabList, 'div', 'right-aligned-tabs').classed('nav ml-auto ', true); - var text = utils$1.sel.safeSelect(div, 'div') - .text(v) - .style('color', textColor) - .style('align-self', 'center'); + plusTab(rightTabs); + sendTab(rightTabs); + closeTab(rightTabs); + var + defaultTab = utils.sel.safeSelect(tabContent, 'div', 'tab-pane') + .classed(utils.str.hypenate(namespace,'default-tab'), true) + .classed("active", true) + .classed('text-left', true), + + p = utils.sel.safeSelect(defaultTab, 'div') + .html( + "Click to add a new group.
"+ + "Click to submit for re-analysis.
"+ + "Click to close the dataselect widget." + ); + // .text('Click the green plus to add a new group.') } - return legend - } - let legends = { - categorical: categoricLegend, numeric: numericLegend - }; + function makeRemoveButton(paneBtnList) { + var + btn = utils.sel.safeSelect(paneBtnList, 'button', 'remove-btn') + .classed('btn btn-danger', true) + .on('click', removeBtnClickToRemove), + i = utils.sel.safeSelect(btn, 'i', 'fa').classed('fa-trash-o', true), + s = utils.sel.safeSelect(btn, 'span').text('Remove group'); + return btn + } + + function removeBtnClickToRemove(d, i) { + var + tab = selection.select('#'+utils.str.hypenate(namespace,'tab',d)), + pane = selection.select('#'+utils.str.hypenate(namespace,'tab','pane',d)); - // import * as d3 from "d3"; - /******************************************************************************* - ** ** - ** ** - ** D3 EXTENSIONS ** - ** ** - ** ** - *******************************************************************************/ - /** - * Recursively ascends parents of selection until it finds an svg tag - * @function d3.selection.thisSVG - * @augments d3.selection - * @returns {Element} which is the svg tag, not the d3 selection of that tag - */ - d3.selection.prototype.thisSVG = function() { return utils$1.sel.getContainingSVG(this.node()); }; + tab.remove(); + pane.remove(); + showRemainingPanes(); + updateTabAndPaneNumbers(); + } - /** - * Helper for getting absolute position of the mouse - * @function d3.mouse.absolute - * @augments d3.mouse - * @returns {number[]} [x, y] as they relate to `html` not to local scope. - */ - d3.mouse.absolute = function() { - var html = d3.select('html').node(); - var [x, y] = this(html); - return [x, y] - }; + function togglePaneById(id) { + var panes = selection.select('.'+utils.str.hypenate(namespace, 'tab-content')); + panes.selectAll('div.tab-pane') + .each(function(d, i){ + d3.select(this).classed('active show', d3.select(this).attr('id') == id); + }); + } - /** - * Gets position of the selection in relation to the containing svg - * @see{@link getContainingSVG} - * @function d3.selection.absolutePosition - * @augments d3.selection - * @returns {Object} with structure similar to getBoundingClientRect, e.g. - * top, left, bottom, right, height, width - */ - d3.selection.prototype.absolutePosition = function() { - var element = this.node(); - var elementPosition = element.getBoundingClientRect(); - var containerSVG = getContainingSVG(element); - var svgPosition = containerSVG.getBoundingClientRect(); + function insertTab(tabs, n) { + + // var li = tabs.insert("li", ':nth-child('+(n)+')') + var li = tabs.insert("li", '.right-aligned-tabs') + .classed('nav-item', true) + .classed('alert', true) + .classed('alert-secondary', true) + .classed('alert-dismissable', true) + .classed('fade', true) + .classed('show', true) + // .classed('mr-auto', true) + .attr("role", 'alert') + .attr("id", utils.str.hypenate(namespace,'tab',n)) + .classed(utils.str.hypenate(namespace,'tab'), true); + + var a = utils.sel.safeSelect(li, 'a') + .attr('data-toggle', 'tab') + .text('Group ' + n) + .attr('href', '#'+utils.str.hypenate(namespace,'tab','pane',n)) + .on('dblclick', function(d, i){ + var t = d3.select(this); + t.attr('contenteditable', true); + d3.select(t.node().parentNode) + .classed('alert-secondary', false) + .classed('alert-warning', true); - return { - top: elementPosition.top - svgPosition.top, - left: elementPosition.left - svgPosition.left, - bottom: elementPosition.bottom - svgPosition.top, - right: elementPosition.right - svgPosition.left, - height: elementPosition.height, - width: elementPosition.width - }; + // d3.select("html").on("click", dispatchBlur) + // + // function dispatchBlur(d, i) { + // if (d3.event.target != d3.select(this)) { + // d3.select(this).dispatch("blur") + // } + // } - }; + }) + .on('blur', function(d, i){ - d3.selection.prototype.relativePositionTo = function(container) { - var element = this.node(); - var elementPosition = element.getBoundingClientRect(); - var containerSVG = container; - var svgPosition = containerSVG.getBoundingClientRect(); + var t = d3.select(this); + t.attr('contenteditable', false); + d3.select(t.node().parentNode) + .classed('alert-secondary', true) + .classed('alert-warning', false); - return { - top: elementPosition.top - svgPosition.top, - left: elementPosition.left - svgPosition.left, - bottom: elementPosition.bottom - svgPosition.top, - right: elementPosition.right - svgPosition.left, - height: elementPosition.height, - width: elementPosition.width - }; + }) + .on('input', function(d, i){ + var t = d3.select(this); + var txt = t.text(); + d3.select(t.attr('href')).select("p.lead").text(txt); + }); - }; - // function getTranslation(selection){ - // var transform = selection.attr('transform') - // var [junk, xy] =transform.split('translate(') - // var [x, y] = xy.split(',') - // y, junk = y.split(')') - // return [parseFloat(x), parseFloat(y)] - // } + return li + } - function lasso( selection ) { - var - svg, // svg that is target of events - objectContainer, // container which houses objects we are selecting (allows for transform to be applied to lasso) - objectClass, // class of object we are selecting - namespace="d3sm-lasso", - chartContainer, - chartOffset, - objectsOffset, - eventCatcher, + function makeTabPane(panes, n) { + var pane = panes.append('div') + .datum(n) + .attr('role', 'tabpanel') + .classed('tab-pane', true) + .classed('text-left', true) + .classed(utils.str.hypenate(namespace,'tab','pane'), true) + .attr('id', utils.str.hypenate(namespace,'tab','pane',n)); + return pane + } - xScale, // optional scale for the lasso currentPoints - yScale, // optional scale for the lasso currentPoints + function populatePane(pane, n) { + var + lead = utils.sel.safeSelect(pane, 'p', 'lead').text('Group '+n), - activeQ = false, // whether or not lasso is active + tableContainer = utils.sel.safeSelect(pane, 'div', 'table-responsive') + .attr("class", utils.str.hypenate(namespace, "data-table")), - currentPoints=[], // mouse points for current lasso - allPoints=[], // list of lists for all points of lassos + table = utils.sel.safeSelect(tableContainer, "table", "table") + .classed("table-sm", true).classed("table-hover", true), - line = d3.line() - .x(function(d, i){ - var x; - if (xScale != undefined) { x = xScale(d[0]); } - else {x = d[0];} - return x //- chartOffset[0]// - objectsOffset[0] - }) - .y(function(d, i){ - var y; - if (yScale != undefined) { y= yScale(d[1]); } - else {y = d[1];} - return y// - chartOffset[1]// - objectsOffset[1] - }) - .curve(d3.curveLinearClosed), + caption = utils.sel.safeSelect(table, "caption", "caption").html("List of selected"), - instance=0, // an indentifier for which instance this lasso is under the current svg + btns = utils.sel.safeSelect(pane, 'div', 'text-right'), - tickDistance = 10, + cN = n % colorFunction$$1.colors().length; + if (n % 2 != 0) { cN = (colorFunction$$1.colors().length - 1) - cN; } - // styles for lasso path - color$$1 = '#17a2b8', - animationRate = '10s', - opacity=0.3, - dashArray = '5, 10', - stroke = 'black', - strokeWidth=2, - // styles for lassoed objects - lassoedFill = "white", - lassoedStroke = 'black', - lassoedStrokeWidth = 3, - transitionDuration = 1000, - easeFunc = d3.easeExp; - var path; - lasso.svg = function(_) { return arguments.length ? (svg = _, lasso) : svg; }; - lasso.chartContainer = function(_) { return arguments.length ? (chartContainer = _, lasso) : chartContainer; }; - lasso.objectContainer = function(_) { return arguments.length ? (objectContainer = _, lasso) : objectContainer; }; - lasso.objectClass = function(_) { return arguments.length ? (objectClass = _, lasso) : objectClass; }; - lasso.namespace = function(_) { return arguments.length ? (namespace = _, lasso) : namespace; }; - lasso.xScale = function(_) { return arguments.length ? (xScale = _, lasso) : xScale; }; - lasso.yScale = function(_) { return arguments.length ? (yScale = _, lasso) : yScale; }; - lasso.activeQ = function(_) { return arguments.length ? (activeQ = _, lasso) : activeQ; }; - lasso.currentPoints = function(_) { return arguments.length ? (currentPoints = _, lasso) : currentPoints; }; - lasso.allPoints = function(_) { return arguments.length ? (allPoints = _, lasso) : allPoints; }; - lasso.instance = function(_) { return arguments.length ? (instance = _, lasso) : instance; }; - lasso.tickDistance = function(_) { return arguments.length ? (tickDistance = _, lasso) : tickDistance; }; - lasso.color = function(_) { return arguments.length ? (color$$1 = _, lasso) : color$$1; }; - lasso.animationRate = function(_) { return arguments.length ? (animationRate = _, lasso) : animationRate; }; - lasso.opacity = function(_) { return arguments.length ? (opacity = _, lasso) : opacity; }; - lasso.dashArray = function(_) { return arguments.length ? (dashArray = _, lasso) : dashArray; }; - lasso.stroke = function(_) { return arguments.length ? (stroke = _, lasso) : stroke; }; - lasso.lassoedFill = function(_) { return arguments.length ? (lassoedFill = _, lasso) : lassoedFill; }; - lasso.lassoedStroke = function(_) { return arguments.length ? (lassoedStroke = _, lasso) : lassoedStroke; }; - lasso.lassoedStrokeWidth = function(_) { return arguments.length ? (lassoedStrokeWidth = _, lasso) : lassoedStrokeWidth; }; - lasso.eventCatcher = function(_) { return arguments.length ? (eventCatcher = _, lasso) : eventCatcher; }; + var lassoBtn = makeLassoButton(btns); + var currentLasso = makeLassoFunction(n, colorFunction$$1.colors()[cN]); + var clearBtn = makeClearButton(btns); - lasso.drag = drag; - lasso.draw = draw; - lasso.tick = tick; - lasso.detect = detect; - lasso.toggle = toggle; - lasso.remove = remove; - lasso.render = render; - lasso.keyFrames = keyFrames; - lasso.updateObjects = updateObjects; - lasso.applyPathAttributes = applyPathAttributes; - lasso.applyObjectAttributes = applyObjectAttributes; + bindButtons(lassoBtn, clearBtn, currentLasso, table); - keyFrames(); + var removeBtn = makeRemoveButton(btns); + + lead.on("mouseover", function(){ + currentLasso.draw(); + }); + lead.on("mouseout", function(){ + currentLasso.remove(); + }); - function lasso() { - // add a dash animation if needed - if (activeQ) { transitionDraw(); } } - function toggle(state) { - // use optional param to set state, otherwise toggle state - activeQ = (state!=undefined) ? state : !activeQ; - chartOffset = utils$1.sel.getTranslation(chartContainer); - objectsOffset = utils$1.sel.getTranslation(objectContainer); + function makeLassoButton(btns) { + var btn = utils.sel.safeSelect(btns, 'button', 'lasso-btn') + .classed('btn btn-info', true) + .classed(namespace, true); + // .datum([lasso]) + var i = utils.sel.safeSelect(btn, 'i', 'fa').classed('fa-hand-pointer-o', true); + var s = utils.sel.safeSelect(btn, 'span').text('Lasso select'); + return btn - if (activeQ) { - svg.node().addEventListener('mousedown', render, true); - } else { - svg.node().removeEventListener('mousedown', render, true); - remove(); - } + } + function makeClearButton(btns) { + var btn = utils.sel.safeSelect(btns, 'button', 'clear-btn') + .classed('btn btn-light', true) + .classed(namespace, true); + var i = utils.sel.safeSelect(btn, 'i', 'fa').classed('fa-eraser', true); + var s = utils.sel.safeSelect(btn, 'span').text('Clear selection'); + return btn } - function draw() { - chartOffset = utils$1.sel.getTranslation(chartContainer); - objectsOffset = utils$1.sel.getTranslation(objectContainer); + function makeLassoFunction(instance, color$$1) { + var currentLasso = lasso() + .namespace(namespace) + .svg(svg) + .objectClass(objectClass) + .chartContainer(chartContainer) + .objectContainer(objectContainer) + .instance(instance) + .color(color$$1) + .yScale(yScale) + .xScale(xScale); + + return currentLasso + } - var container = utils$1.sel.safeSelect(objectContainer, 'g', 'lasso-container'); - var paths$$1 = container.selectAll('path[instance="'+instance+'"]'); + function bindButtons(lassoBtn, clearBtn, currentLasso, table) { + currentLasso.eventCatcher(lassoBtn); + lassoBtn.node().addEventListener("click", lassoBtnToggle); + + lassoBtn.datum([currentLasso]) + .on(utils.str.hypenate(namespace,'render'), function(){ + d3.select(this).datum()[0].draw(); + }) + .on(utils.str.hypenate(namespace,"drag"), function(las){ + var nodes = chartContainer.selectAll(".in-lasso-"+las[0].instance()).nodes(); + var tableData = nodes.map(function(d, i){ + var extracted = dataExtractor(d3.select(d).datum()); + extracted["__node"] = d; + return extracted + }); - // update - paths$$1 = paths$$1.data(allPoints); - // remove excess - var pExit = paths$$1.exit().remove(); - // add needed paths - var pEnter = paths$$1.enter().append('path'); + makeTable(table, tableData, currentLasso); + }); - // merge - paths$$1 = paths$$1.merge(pEnter); + clearBtn.on('click', function(){ + currentLasso.allPoints([]); + currentLasso.currentPoints([]); + lassoBtn.dispatch(utils.str.hypenate(namespace,"drag")); - // apply - applyPathAttributes(paths$$1); - } + }); - function remove() { - var container = utils$1.sel.safeSelect(objectContainer, 'g', 'lasso-container'); - var paths$$1 = container.selectAll('path[instance="'+instance+'"]').remove(); - container.remove(); - objectContainer.selectAll(objectClass).classed("in-lasso", false); - updateObjects(); } - function render( event ) { - // nothing can interefer with drawing the lasso - event.preventDefault(); event.stopPropagation(); + function lassoBtnToggle() { + var that = d3.select(this); + var las = that.datum()[0]; - var container = utils$1.sel.safeSelect(objectContainer, 'g', 'lasso-container'); + las.toggle(); + var activeQ = las.activeQ(); - /* - each time the user presses down, while the state is active, the lasso - the lasso should make a seperate segment. - */ - currentPoints = []; + if (las.activeQ()) { + that.classed('btn-info', !activeQ); + that.classed('btn-warning', activeQ); + that.select("span").text("Lasso select (active)"); + selection.selectAll("."+namespace+".lasso-btn").dispatch(utils.str.hypenate(namespace,'render')); + d3.select("html").node().addEventListener('mousedown', monitorLassoButtonState); + } else { + that.classed('btn-info', !activeQ); + that.classed('btn-warning', activeQ); + that.select("span").text("Lasso select"); + d3.select("html").node().removeEventListener('mousedown', monitorLassoButtonState); + } - svg.node().addEventListener('mousemove', drag); - svg.node().addEventListener('mouseup', function(event) { - svg.node().removeEventListener('mousemove', drag); - allPoints.push(currentPoints); - // BUG: somehow this is pushing currentPoints n times where n is the nth lasso path for the current instance - // NOTE: allPoints = utils.arr.unique(allPoints) is a temporary and inefficient fix - allPoints = utils$1.arr.unique(allPoints); - }); + function monitorLassoButtonState(event) { + /* + activeLasso stops event stopPropagation, so this event will not register in + that case. Thus we only need to ensure that the usre is clicking on the lasso button + otherwise, de-spawn lasso. + */ + if ( + event.target != that.node() && + event.target != that.select("span").node() && + event.target != that.select("i").node() + ) { + // event.preventDefault() + // event.stopPropagation() + las.toggle(false); + that.classed('btn-info', true); + that.classed('btn-warning', false); + that.select("span").text("Lasso select"); + // console.log(that, that.node()) + // that.dispatch("focusout") + } + } - path = container.append('path').data([currentPoints]); - applyPathAttributes(path); } - function transitionDraw() { - var container = utils$1.sel.safeSelect(objectContainer, 'g', 'lasso-container'); - var paths$$1 = container.selectAll('path[instance="'+instance+'"]'); - // update - paths$$1 = paths$$1.data(allPoints); + function updateTableHeaderColumns(headR, tableData) { + var headerKeys = d3.keys(tableData[0]).filter(k=>k!="__node"); + if (headerKeys.length > 0) { + // headerKeys = ["remove"].concat(headerKeys) + headerKeys.push("remove"); + } - // remove excess - var pExit = paths$$1.exit().remove(); - // add needed paths - var pEnter = paths$$1.enter().append('path'); + var headerCols = headR.selectAll("th"); - // merge - paths$$1 = paths$$1.merge(pEnter) - .transition().duration(transitionDuration) - .ease(easeFunc); - applyPathAttributes(paths$$1); + headerCols = headerCols.data(headerKeys); + headerCols.exit().remove(); + headerCols = headerCols.merge(headerCols.enter().append("th").attr("scope","col")) + .text(function(d, i){return d}); + return headerKeys } - function applyPathAttributes(path) { - path - .attr("class", utils$1.str.hypenate(namespace, "lasso-path")) - .style('opacity', opacity) - .attr('fill', color$$1) - .attr("d", line) - .attr('instance', instance) - .style("stroke-dasharray", dashArray) - .attr("stroke", stroke) - .attr("stroke-width", strokeWidth) - .style('animation', 'lassoDash '+animationRate+' linear') - .style("animation-iteration-count", "infinite"); + function updateTableRows(body, tableData) { + var bodyRows = body.selectAll("tr"); + + bodyRows = bodyRows.data(tableData); + bodyRows.exit().remove(); + bodyRows = bodyRows.merge(bodyRows.enter().append("tr")); + + return bodyRows } - function drag(event) { - /* - effectively create a mouse down and move event (which normally is inteperated - as 'drag' by the browser) by dynamically adding / removing this event on - mouse down / mouse up. - */ + function updateTableRowColumns(cols, headerKeys, rowData, tableData, lasso$$1, table) { + cols = cols.data(headerKeys); + cols.exit().remove(); + cols = cols.merge(cols.enter().append("td")); + + cols.html(function(d, i){ + if (d != "remove") { return rowData[d] } + return "" + }).classed("text-left", function(d, i){ + if (d != "remove") { return false } + return true + }); + // .attr("scope",function(d, i){ + // if (d != "remove") { return false } + // return true + // }) + + cols.select("i.fa-close").on("click", function(d, i){ + var node = rowData["__node"], + n = d3.select(node); + n.classed("in-lasso", false); + n.classed("in-lasso-"+lasso$$1.instance(), false); + + tableData.map(function(dd, j){ + if (dd["__node"] == node) { + tableData.splice(j, 1); + makeTable(table, tableData, lasso$$1); + } + }); + + }); + } + + function makeTable(table, tableData, lasso$$1) { + var + head = utils.sel.safeSelect(table, "thead"), + headR = utils.sel.safeSelect(head, "tr"), + body = utils.sel.safeSelect(table, "tbody"), + + headerKeys = updateTableHeaderColumns(headR, tableData), - if (eventCatcher != undefined) {eventCatcher.dispatch(utils$1.str.hypenate(namespace,"drag"));} - // d3.dispatch(utils.str.hypenate(namespace,"drag")) + bodyRows = updateTableRows(body, tableData); - if (event.which != 1) {return} // ensures left mouse button set - d3.event = event; - var pt = d3.mouse(objectContainer.node()); - var pt = d3.mouse(svg.node()); + bodyRows.each(function(rowData, i){ + var t = d3.select(this); + var cols = t.selectAll("td"); - if (xScale != undefined) {pt[0] = xScale.invert(pt[0]);} - if (yScale != undefined) {pt[1] = yScale.invert(pt[1]);} - pt[0] = pt[0] - chartOffset[0] - objectsOffset[0]; - pt[1] = pt[1] - chartOffset[1] - objectsOffset[1]; + updateTableRowColumns(cols, headerKeys, rowData, tableData, lasso$$1, table); - /* if we have a point already, test if it passes a minimum distance to prevent overwhelming with too many tick functions */ - if (currentPoints.length) { - var lastPt = currentPoints[currentPoints.length - 1]; - var a = [pt[0], pt[1]], b = [lastPt[0], lastPt[1]]; + t.on("mouseover", function(d, i) { + lasso$$1.applyObjectAttributes(d3.select(d["__node"]),true); + }) + .on("mouseout", function(d, i) { + lasso$$1.applyObjectAttributes(d3.select(d["__node"]),false); + }); + }); - if (xScale) {b[0] = xScale(b[0]); a[0] = xScale(a[0]);} - if (yScale) {b[1] = yScale(b[1]); a[1] = yScale(a[1]);} - var dist = utils$1.math.euclideanDistance(b, a); - if (dist > tickDistance) { tick(pt); } - } - else { tick(pt); } } - function tick (pt) { - /* - If a point is provided update data and objects. - Otherwise just call on data we already have. - Why like this?: - 1. currentPoints is current points to allow disjunct lassos, currentPoints is only pushed to - allPoints after mouseup. - 2. to allow render of objects in the lasso class / updating the data list - just by toggling the button - */ - if (pt != undefined) { - currentPoints.push(pt); - path.attr("d", line); - if (currentPoints.length < 3) {return} // need at least 3 points to detect anything. - detect(allPoints.concat([currentPoints])); - } else { - detect(allPoints); - } - } - function detect(lassos) { - if (lassos == undefined) {lassos = allPoints;} - objectContainer.selectAll(objectClass).each(function(d, i){ - var current = d3.select(this), - box = current.absolutePosition(), - // box = current.relativePositionTo(objectContainer.node()), + function makeNewGroup(d, i) { - boxPts = [ - [ - box.left - chartOffset[0] - objectsOffset[0], - box.top - chartOffset[1] - objectsOffset[1] - ], - [ - box.right - chartOffset[0] - objectsOffset[0], - box.top - chartOffset[1] - objectsOffset[1] - ], - [ - box.left - chartOffset[0] - objectsOffset[0], - box.bottom - chartOffset[1] - objectsOffset[1] - ], - [ - box.right - chartOffset[0] - objectsOffset[0], - box.bottom - chartOffset[1] - objectsOffset[1] - ] - ]; + var tabs = selection.select('.'+utils.str.hypenate(namespace, 'tab-list')), + panes = selection.select('.'+utils.str.hypenate(namespace, 'tab-content')), + n = newGroupNumber(); - if (xScale != undefined) { - boxPts[0][0] = xScale.invert(boxPts[0][0]); - boxPts[1][0] = xScale.invert(boxPts[1][0]); - boxPts[2][0] = xScale.invert(boxPts[2][0]); - boxPts[3][0] = xScale.invert(boxPts[3][0]); - } - if (yScale != undefined) { - boxPts[0][1] = yScale.invert(boxPts[0][1]); - boxPts[1][1] = yScale.invert(boxPts[1][1]); - boxPts[2][1] = yScale.invert(boxPts[2][1]); - boxPts[3][1] = yScale.invert(boxPts[3][1]); - } + if (tabs.selectAll('.'+utils.str.hypenate(namespace,'tab')).size() == maxNumberOfGroups) { + onError('only '+maxNumberOfGroups+' allowed.', 'warning'); + return + } + var + tab = insertTab(tabs, n), + pane = makeTabPane(panes, n); - /* - flag needed as we have to test multiple lasso segments, and if the point - is not in one segment, it does not mean it is not in any - */ - var inAnyLassoQ = false; - for (var i = 0; i < lassos.length; i++) { - var lassoPoints = lassos[i]; - // .map(function(pt){ - // var x, y = pt - // if (xScale!=undefined) {x = xScale(x)} - // if (yScale!=undefined) {y = yScale(y)} - // return [x, y] - // }) - var boxInLassoQ = boxPts.every(coord => d3.polygonContains(lassoPoints, coord)); + populatePane(pane, n); + togglePaneById(pane.attr('id')); - if (boxInLassoQ) { inAnyLassoQ = true; } // only update flag in the positive case. - } + } - current.classed('in-lasso', inAnyLassoQ); - current.classed('in-lasso-'+instance, inAnyLassoQ); + function newGroupNumber() { + var tabs = selection.select('.'+utils.str.hypenate(namespace, 'tab-list'));//, + var + n = tabs.selectAll('li').size() - 3, // minus 1 for plus tab + s = 'Group ' + n, + gs = []; + tabs.each(function(d, i){ return gs.push(d3.select(this).select("a").text()) }); + while (gs.includes(s)) { n+=1; s = 'Group ' + n; } + return n + } + + function updateTabAndPaneNumbers() { + var + tabs = selection.select('.'+utils.str.hypenate(namespace, 'tab-list')), + panes = selection.select('.'+utils.str.hypenate(namespace, 'tab-content')); + + tabs = tabs.selectAll('.'+utils.str.hypenate(namespace,'tab')); + panes = panes.selectAll('.'+utils.str.hypenate(namespace,'tab', 'pane')); + + tabs.each(function(d, i){ + d3.select(this).datum(i) + .attr("id", utils.str.hypenate(namespace,'tab',i)) + .select('a') + .attr('href', '#'+utils.str.hypenate(namespace,'tab','pane',i)) + .text(function(dd, ii){ + var curText = d3.select(this).text(); + if (curText.split(' ')[0] == 'Group') { + return 'Group ' + i + } + return curText + }); + }); + + panes.each(function(d, i){ + d3.select(this).datum(i) + .attr('id', utils.str.hypenate(namespace,'tab','pane',i)); + utils.sel.safeSelect(d3.select(this), 'p', 'lead') + .text(function(dd, ii){ + var curText = d3.select(this).text(); + if (curText.split(' ')[0] == 'Group') { + return 'Group ' + i + } + return curText + }); + utils.sel.safeSelect(d3.select(this), 'button', 'remove-btn') + .on('click', removeBtnClickToRemove); }); - updateObjects(); - return objectContainer.selectAll('.in-lasso-'+instance) } + function gatherDataLists() { + var tabs = selection.select('.'+utils.str.hypenate(namespace, 'tab-list')); + var panes = selection.select('.'+utils.str.hypenate(namespace, 'tab-content')); + var tables = panes.selectAll('.'+utils.str.hypenate(namespace, "data-table")); + var data = {}; + var textGroups = tabs.selectAll('li.'+utils.str.hypenate(namespace,'tab') + ' > a') + .nodes().map(function(d, i){return d3.select(d).text()}); - function updateObjects() { - objectContainer.selectAll(objectClass).each(function(d, i) { - var t = d3.select(this); - applyObjectAttributes(t, t.classed('in-lasso')); + textGroups.map(function(e, i){ + data[e] = []; }); - } - function applyObjectAttributes(obj, setQ) { - var - preLassoFill = obj.attr('_pre_lasso_fill'), - preLassoStroke = obj.attr('_pre_lasso_stroke'), - preLassoStrokeWidth = obj.attr('_pre_lasso_stroke-width'); - if (setQ) { - obj.classed("in-lasso", true); - obj.classed('in-lasso-'+instance, true); - if (preLassoFill == undefined) { obj.attr('_pre_lasso_fill', obj.attr('fill')); } - if (preLassoStroke == undefined) { obj.attr('_pre_lasso_stroke', obj.attr('stroke')); } - if (preLassoStrokeWidth == undefined) { obj.attr('_pre_lasso_stroke-width', obj.attr('stroke-width')); } + tables.each(function(d, i){ + var table = d3.select(this).select("tbody"); + data[textGroups[i]] = table.selectAll('tr').data(); + }); - obj - //BUG: when .raise() - .attr('fill', lassoedFill) - .attr('stroke', lassoedStroke) - .attr('stoke-width', lassoedStrokeWidth); + return data + } - } else { - obj.classed("in-lasso", false); - obj.classed('in-lasso-'+instance, false); - if (preLassoFill != undefined) { obj.attr('fill', preLassoFill); } - if (preLassoStroke != undefined) { obj.attr('stroke', preLassoStroke); } - if (preLassoStrokeWidth != undefined) { obj.attr('stroke-width', preLassoStrokeWidth); } + function showRemainingPanes() { + var + tabs = selection.select('.'+utils.str.hypenate(namespace, 'tab-list')), + panes = selection.select('.'+utils.str.hypenate(namespace, 'tab-content')), + remainingTabs = tabs.selectAll('.'+utils.str.hypenate(namespace,'tab')); + + if (remainingTabs.size() == 0) { + panes.select('.'+utils.str.hypenate(namespace,'default-tab')) + .classed("active", true) + .classed('text-left', true); + } + else { + var lastTab = remainingTabs.nodes()[remainingTabs.size()-1], + lastPaneId = d3.select(lastTab).select('a').attr('href'); + panes.select(lastPaneId) + .classed("active", true) + .classed('text-left', true); } } - function keyFrames() { - var style = - d3.select("html").select('style.'+utils$1.str.hypenate(namespace,"lasso-dash")); - if (style.empty()) { - d3.select("html").append('style') - .classed(utils$1.str.hypenate(namespace,"lasso-dash"), true) - .html("@keyframes lassoDash {to { stroke-dashoffset: 1000;}}"); - } - } - return lasso + return lassoWidget } /******************************************************************************* @@ -8070,14 +9392,14 @@ function setLocks() { - var chartObjSel = chartSel.select('.'+utils$1.str.hypenate(chart.namespace(),'object-container')); - var chartObjTrans = utils$1.sel.getTranslation(chartObjSel.attr('transform')); + var chartObjSel = chartSel.select('.'+utils.str.hypenate(chart.namespace(),'object-container')); + var chartObjTrans = utils.sel.getTranslation(chartObjSel.attr('transform')); var cos = chartObjSel.attr('transform', 'translate(0,0)'); xLock = chartSel.node().getBBox().width - chart.spaceX();// * .9 yLock = chartSel.node().getBBox().height - chart.spaceY();// * .9 cos.attr('transform', 'translate('+chartObjTrans[0]+','+chartObjTrans[1]+')'); - utils$1.con.log('plotZoom', 'setLocks', {xLock:xLock, yLock:yLock}); + utils.con.log('plotZoom', 'setLocks', {xLock:xLock, yLock:yLock}); } @@ -8137,20 +9459,20 @@ - var chartObjSel = chartSel.select('.'+utils$1.str.hypenate(chart.namespace(),'object-container')); + var chartObjSel = chartSel.select('.'+utils.str.hypenate(chart.namespace(),'object-container')); var xComponentObjSel = xComponentsSel.map(function(d, i){ - return d.select('.'+utils$1.str.hypenate(xComponents[i].namespace(),'object-container')) + return d.select('.'+utils.str.hypenate(xComponents[i].namespace(),'object-container')) }); var yComponentObjSel = yComponentsSel.map(function(d, i){ - return d.select('.'+utils$1.str.hypenate(yComponents[i].namespace(),'object-container')) + return d.select('.'+utils.str.hypenate(yComponents[i].namespace(),'object-container')) }); - var chartObjTrans = utils$1.sel.getTranslation(chartObjSel.attr('transform')); + var chartObjTrans = utils.sel.getTranslation(chartObjSel.attr('transform')); var xComponentsObjTrans = xComponentObjSel.map(function(d, i){ - return utils$1.sel.getTranslation(d.attr('transform')) + return utils.sel.getTranslation(d.attr('transform')) }); var yComponentsObjTrans = yComponentObjSel.map(function(d, i){ - return utils$1.sel.getTranslation(d.attr('transform')) + return utils.sel.getTranslation(d.attr('transform')) }); var x = horizontalQ ? transform.applyX(chartObjTrans[0]) : 0; @@ -8175,9 +9497,9 @@ zoom.reset = function() { - var chartObjSel = chartSel.select('.'+utils$1.str.hypenate(chart.namespace(),'object-container')); - var xAxisObjSel = xAxisSel.select('.'+utils$1.str.hypenate(xAxis.namespace(),'object-container')); - var yAxisObjSel = yAxisSel.select('.'+utils$1.str.hypenate(yAxis.namespace(),'object-container')); + var chartObjSel = chartSel.select('.'+utils.str.hypenate(chart.namespace(),'object-container')); + var xAxisObjSel = xAxisSel.select('.'+utils.str.hypenate(xAxis.namespace(),'object-container')); + var yAxisObjSel = yAxisSel.select('.'+utils.str.hypenate(yAxis.namespace(),'object-container')); chartObjSel.attr('transform', 'translate('+0+','+0+')'); xAxisObjSel.attr('transform', 'translate('+0+','+0+')'); yAxisObjSel.attr('transform', 'translate('+0+','+0+')'); @@ -8308,13 +9630,13 @@ function setLocks() { - var chartObjSel = chartSel.select('.'+utils$1.str.hypenate(chart.namespace(),'object-container')); - var chartObjTrans = utils$1.sel.getTranslation(chartObjSel.attr('transform')); + var chartObjSel = chartSel.select('.'+utils.str.hypenate(chart.namespace(),'object-container')); + var chartObjTrans = utils.sel.getTranslation(chartObjSel.attr('transform')); var cos = chartObjSel.attr('transform', 'translate(0,0)'); xLock = chartSel.node().getBBox().width - chart.spaceX() * .9; yLock = chartSel.node().getBBox().height - chart.spaceY() * .9; cos.attr('transform', 'translate('+chartObjTrans[0]+','+chartObjTrans[1]+')'); - utils$1.con.log('plotZoom', 'setLocks', {xLock:xLock, yLock:yLock}); + utils.con.log('plotZoom', 'setLocks', {xLock:xLock, yLock:yLock}); } @@ -8373,9 +9695,9 @@ - var chartObjSel = chartSel.select('.'+utils$1.str.hypenate(chart.namespace(),'object-container')); - var xAxisObjSel = xAxisSel.select('.'+utils$1.str.hypenate(xAxis.namespace(),'object-container')); - var yAxisObjSel = yAxisSel.select('.'+utils$1.str.hypenate(yAxis.namespace(),'object-container')); + var chartObjSel = chartSel.select('.'+utils.str.hypenate(chart.namespace(),'object-container')); + var xAxisObjSel = xAxisSel.select('.'+utils.str.hypenate(xAxis.namespace(),'object-container')); + var yAxisObjSel = yAxisSel.select('.'+utils.str.hypenate(yAxis.namespace(),'object-container')); // xLock = chartSel.node().getBBox().width - chart.spaceX() - chartSel.node().getBBox().x // yLock = chartSel.node().getBBox().height - chart.spaceY() @@ -8383,9 +9705,9 @@ // bhm.selection().node().getBBox().width - bhm.spaceX() - var chartObjTrans = utils$1.sel.getTranslation(chartObjSel.attr('transform')); - var xAxisObjTrans = utils$1.sel.getTranslation(xAxisObjSel.attr('transform')); - var yAxisObjTrans = utils$1.sel.getTranslation(yAxisObjSel.attr('transform')); + var chartObjTrans = utils.sel.getTranslation(chartObjSel.attr('transform')); + var xAxisObjTrans = utils.sel.getTranslation(xAxisObjSel.attr('transform')); + var yAxisObjTrans = utils.sel.getTranslation(yAxisObjSel.attr('transform')); var x = horizontalQ ? transform.applyX(chartObjTrans[0]) : 0; @@ -8407,9 +9729,9 @@ zoom.reset = function() { - var chartObjSel = chartSel.select('.'+utils$1.str.hypenate(chart.namespace(),'object-container')); - var xAxisObjSel = xAxisSel.select('.'+utils$1.str.hypenate(xAxis.namespace(),'object-container')); - var yAxisObjSel = yAxisSel.select('.'+utils$1.str.hypenate(yAxis.namespace(),'object-container')); + var chartObjSel = chartSel.select('.'+utils.str.hypenate(chart.namespace(),'object-container')); + var xAxisObjSel = xAxisSel.select('.'+utils.str.hypenate(xAxis.namespace(),'object-container')); + var yAxisObjSel = yAxisSel.select('.'+utils.str.hypenate(yAxis.namespace(),'object-container')); chartObjSel.attr('transform', 'translate('+0+','+0+')'); xAxisObjSel.attr('transform', 'translate('+0+','+0+')'); yAxisObjSel.attr('transform', 'translate('+0+','+0+')'); @@ -8418,11 +9740,297 @@ return zoom } + function selectFilter(selection) { + + var + data, + namespace = 'd3sm-select-filter', + selectionName = 'Select options:', + defaultValue = undefined; + + + + + var lastValue = undefined; + + selectFilter.data = function(_) { return arguments.length ? (data = _, selectFilter) : data}; + selectFilter.namespace = function(_) { return arguments.length ? (namespace = _, selectFilter) : namespace}; + selectFilter.selectionName = function(_) { return arguments.length ? (selectionName = _, selectFilter) : selectionName}; + selectFilter.defaultValue = function(_) { return arguments.length ? (defaultValue = _, selectFilter) : defaultValue}; + selectFilter.currentOption = currentOption; + + function selectFilter() { + var + container = utils.sel.safeSelect(selection, 'div', 'input-group').classed(utils.str.hypenate(namespace,'container'),true), + + selectPrepend = utils.sel.safeSelect(container, 'div', 'select-prepend').classed('input-group-prepend', true), + selectPrependSpan = utils.sel.safeSelect(selectPrepend, 'span', 'input-group-text').text(selectionName), + + select = utils.sel.safeSelect(container, 'select', 'custom-select').classed(utils.str.hypenate(namespace,'select'),true), + + selectAppend = utils.sel.safeSelect(container, 'div', 'select-append').classed('input-group-prepend', true), + selectAppendButton = utils.sel.safeSelect(selectAppend, 'a', 'filter-button').classed('btn btn-outline-secondary', true), + filterButtonIcon = utils.sel.safeSelect(selectAppendButton, 'i', 'fa fa-filter'), + + inputGroup = utils.sel.safeSelect(container, 'div', 'filter-input-group').classed('input-group',true).classed('d-none', true), + inputPrepend = utils.sel.safeSelect(inputGroup, 'div', 'input-group-prepend'), + inputPrependSpan = utils.sel.safeSelect(inputPrepend, 'span', 'input-group-text').classed('search-button', true), + inputPrependSpanIcon = utils.sel.safeSelect(inputPrependSpan,'i','fa fa-search'), + + input = utils.sel.safeSelect(inputGroup, 'input', 'form-control').attr('placeholder', 'all').attr('type', 'text'), + inputAppend = utils.sel.safeSelect(inputGroup, 'div', 'input-group-append'), + inputAppendButton = utils.sel.safeSelect(inputAppend, 'a', 'close-button').classed('btn btn-outline-secondary', true), + inputAppendButtonIcon = utils.sel.safeSelect(inputAppendButton, 'i', 'fa fa-close'); + + + var keys = d3.keys(data), + options = select.selectAll('option'); + + options = options.data(d3.keys(data)); + options = options.merge(options.enter().append('option')) + .attr('value', function(d, i){return d}) + .text(function(d, i){return d}); + + var + filterButton = selectAppendButton, + closeButton = inputAppendButton; + + filterButton.on('click', function(d, i){ + var currentStyle = inputGroup.classed('d-none'); + inputGroup.classed('d-none', !currentStyle); + }); + + closeButton.on('click', function(d, i){ + input.property('value', '').dispatch('input'); + }); + + input.on('input', function(d, i){ + var + val = input.property('value'), + reg = new RegExp(val, 'gi'), + use; + + if (val == '') {use = keys;} + else { + use = []; + d3.keys(data).map(function(option, j){ + var match = option.match(reg); + if (match == null || match.join('') == '') ; + else { use.push(option); } + }); + } + + options = select.selectAll('option'); + options = options.data(use); + options.exit().remove(); + options = options.merge(options.enter().append('option')) + .attr('value', function(d, i){return d}) + .text(function(d, i){return d}); + + var current = currentOption(); + if (lastValue != current) { + lastValue = current; + select.dispatch('change'); + } + }); + + + } + + function currentOption() { + var val = selection.select("select").property('value'); + return val == undefined || val == '' + ? defaultValue == undefined + ? d3.keys(data)[0] + : defaultValue + : val + } + + return selectFilter + } + + /******************************************************************************* + ** ** + ** ** + ** DATATOGGLE ** + ** ** + ** ** + *******************************************************************************/ + /** + * Creates a datatoggle + * @constructor datatoggle + * @param {d3.selection} selection + * @namespace datatoggle + * @returns {function} datatoggle + */ + function datatoggle( selection ) { + var + /** + * Keys to make toggle-able options + * (see {@link datatoggle#keys}) + * @param {string[]} [keys=undefined] + * @memberof datatoggle# + * @property + */ + keys, + + /** + * What to do when a different key is clicked + * (see {@link datatoggle#updateFunction}) + * @param {function} [updateFunction=function(){}] + * @memberof datatoggle# + * @property + */ + updateFunction = function(){}, + /** + * Namespace for all items made by this instance of datatoggle + * @param {string} [namespace="d3sm-databar"] + * @memberof datatoggle# + * @property + */ + namespace='d3sm-databar', + /** + * Currently toggled key + * @param {string} [currentKey=undefined] + * @memberof datatoggle# + * @property + */ + currentKey, + + + + xAxisSelectQ = false, + xAxisOptions, + yAxisSelectQ = false, + yAxisOptions, + data; + toggle.xAxisSelectQ = function(_){return arguments.length ? (xAxisSelectQ = _, toggle) : xAxisSelectQ}; + toggle.yAxisSelectQ = function(_){return arguments.length ? (yAxisSelectQ = _, toggle) : yAxisSelectQ}; + toggle.xAxisOptions = function(_){return arguments.length ? (xAxisOptions = _, toggle) : xAxisOptions}; + toggle.yAxisOptions = function(_){return arguments.length ? (yAxisOptions = _, toggle) : yAxisOptions}; + toggle.data = function(_){return arguments.length ? (data = _, toggle) : data}; + toggle.keys = function(_){return arguments.length ? (keys = _, toggle) : keys}; + toggle.currentKey = function(_){return arguments.length ? (currentKey = _, toggle) : currentKey}; + + /** + * Gets / sets the updateFunction + * (see {@link datatoggle#updateFunction}) + * @function datatoggle.updateFunction + * @param {function} [_=none] + * @returns {datatoggle | function} + * @memberof datatoggle + * @property + * by default updateFunction = function(){} + */ + toggle.updateFunction = function(_){return arguments.length ? (updateFunction = _, toggle) : updateFunction}; + /** + * Gets / sets the namespace + * (see {@link datatoggle#namespace}) + * @function datatoggle.namespace + * @param {string} [_=none] + * @returns {datatoggle | string} + * @memberof datatoggle + * @property + * by default namespace = 'd3sm-databar' + */ + toggle.namespace = function(_){return arguments.length ? (namespace = _, toggle) : namespace}; + /** + * Gets / sets the currentKey + * (see {@link datatoggle#currentKey}) + * @function datatoggle.currentKey + * @param {string} [_=none] + * @returns {datatoggle | string} + * @memberof datatoggle + * @property + * by default currentKey = undefined + */ + toggle.currentKeys = + function () { + var vals = {}; + d3.keys(filterSelects).map(function(k, i){ + vals[k]= filterSelects[k].currentOption(); + }); + return vals + }; + + var filterSelects = {}; + function toggle() { + // selection options + + // selection.classed('d-flex flex-row', true) + // var filterButton = utils.sel.safeSelect(selection, 'a', 'slider-buttons') + // var filterI = utils.sel.safeSelect(filterButton, 'i', 'fa fa-sliders') + + /*BUG: unexpected behavior. + - when using bootstrap-eque way for collapse, clicking button submits to + the same page in applications (but not in demo), so using anchor (
) + - when using anchor, cause page to jump to that location + - when using show, the first open works, but the close does not. + */ + // filterButton.attr('class', 'btn btn-secondary') + // .attr('data-toggle', 'collapse') + // .attr('href', '#'+utils.str.hypenate(namespace, 'data-toggle')) + // .attr('target', '_blank') + // .html(filterButton.html()+' Filters') + // .on('click', function(d, i){ + // d3.event.preventDefault() + // d3.event.stopPropagation() + // var dt = d3.select("#"+utils.str.hypenate(namespace, 'data-toggle')) + // dt.classed('show', !dt.classed('show')) + // dt.classed('d-inline-flex', dt.classed('show')) + // + // filterButton.classed('btn-primary', dt.classed('show')) + // filterButton.classed('btn-secondary', !dt.classed('show')) + // }) + // .classed('d-inline-flex', true) + // .style("margin-right", '10px') + + + + // var datatoggleCollapse = utils.sel.safeSelect(selection, 'div', utils.str.hypenate(namespace,'collapse')) + // .attr('id', utils.str.hypenate(namespace, 'data-toggle')) + // .classed('collapse', true) + + // var flexRow = utils.sel.safeSelect(datatoggleCollapse, 'div', 'd-inline-flex flex-row flex-wrap') + var flexRow = utils.sel.safeSelect(selection, 'div', 'd-inline-flex flex-row flex-wrap'); + + var dataopts = flexRow.selectAll('div.'+utils.str.hypenate(namespace,'select-filter')); + // remove excess + dataopts.exit().remove(); + // bind data + dataopts = dataopts.data(d3.keys(data)); + //enter + var doEnter = dataopts.enter().append('div') + .attr('class', 'select-filter'); + + dataopts = dataopts.merge(doEnter).style('margin-right', "10px"); + + dataopts.each(function(d, i){ + var t = d3.select(this); + var sf = selectFilter(t) + .data(data[d]) + .namespace(utils.str.hypenate(namespace, d)) + .selectionName(d); + sf(); + filterSelects[d] = sf; + }); + + + selection.selectAll('select') + .on('change', function(){updateFunction();}); + // bind update function + // d3.selectAll + return toggle + } + + return toggle + } + let aux = { - lasso, plotZoom, multiPlotZoom + lasso, plotZoom, multiPlotZoom, datatoggle, selectFilter, lassoWidget }; - let d3sm$1 = { + let d3sm = { aux, axis, charts, @@ -8431,14 +10039,14 @@ groupingSpacer, legends, tooltip, - utils: utils$1, + utils, }; if (typeof window !== 'undefined') { - window.d3sm = d3sm$1; + window.d3sm = d3sm; } - exports.default = d3sm$1; + exports.default = d3sm; Object.defineProperty(exports, '__esModule', { value: true }); diff --git a/index.html b/index.html index 7d3527d..8a08d16 100644 --- a/index.html +++ b/index.html @@ -4,10 +4,10 @@ d3sm - + - - + + diff --git a/src/modules/aux/data-toggle.js b/src/modules/aux/data-toggle.js new file mode 100644 index 0000000..be2e016 --- /dev/null +++ b/src/modules/aux/data-toggle.js @@ -0,0 +1,210 @@ +import utils from '../utils/index.js' +import selectFilter from './select-filter'; + +/******************************************************************************* +** ** +** ** +** DATATOGGLE ** +** ** +** ** +*******************************************************************************/ +/** + * Creates a datatoggle + * @constructor datatoggle + * @param {d3.selection} selection + * @namespace datatoggle + * @returns {function} datatoggle + */ +export default function datatoggle( selection ) { + var + /** + * Keys to make toggle-able options + * (see {@link datatoggle#keys}) + * @param {string[]} [keys=undefined] + * @memberof datatoggle# + * @property + */ + keys, + + /** + * What to do when a different key is clicked + * (see {@link datatoggle#updateFunction}) + * @param {function} [updateFunction=function(){}] + * @memberof datatoggle# + * @property + */ + updateFunction = function(){}, + /** + * Namespace for all items made by this instance of datatoggle + * @param {string} [namespace="d3sm-databar"] + * @memberof datatoggle# + * @property + */ + namespace='d3sm-databar', + /** + * Currently toggled key + * @param {string} [currentKey=undefined] + * @memberof datatoggle# + * @property + */ + currentKey, + + + + xAxisSelectQ = false, + xAxisOptions, + yAxisSelectQ = false, + yAxisOptions, + data + toggle.xAxisSelectQ = function(_){return arguments.length ? (xAxisSelectQ = _, toggle) : xAxisSelectQ} + toggle.yAxisSelectQ = function(_){return arguments.length ? (yAxisSelectQ = _, toggle) : yAxisSelectQ} + toggle.xAxisOptions = function(_){return arguments.length ? (xAxisOptions = _, toggle) : xAxisOptions} + toggle.yAxisOptions = function(_){return arguments.length ? (yAxisOptions = _, toggle) : yAxisOptions} + toggle.data = function(_){return arguments.length ? (data = _, toggle) : data} + toggle.keys = function(_){return arguments.length ? (keys = _, toggle) : keys} + toggle.currentKey = function(_){return arguments.length ? (currentKey = _, toggle) : currentKey} + + /** + * Gets / sets the updateFunction + * (see {@link datatoggle#updateFunction}) + * @function datatoggle.updateFunction + * @param {function} [_=none] + * @returns {datatoggle | function} + * @memberof datatoggle + * @property + * by default updateFunction = function(){} + */ + toggle.updateFunction = function(_){return arguments.length ? (updateFunction = _, toggle) : updateFunction} + /** + * Gets / sets the namespace + * (see {@link datatoggle#namespace}) + * @function datatoggle.namespace + * @param {string} [_=none] + * @returns {datatoggle | string} + * @memberof datatoggle + * @property + * by default namespace = 'd3sm-databar' + */ + toggle.namespace = function(_){return arguments.length ? (namespace = _, toggle) : namespace} + /** + * Gets / sets the currentKey + * (see {@link datatoggle#currentKey}) + * @function datatoggle.currentKey + * @param {string} [_=none] + * @returns {datatoggle | string} + * @memberof datatoggle + * @property + * by default currentKey = undefined + */ + toggle.currentKeys = + function () { + var vals = {} + d3.keys(filterSelects).map(function(k, i){ + vals[k]= filterSelects[k].currentOption() + }) + return vals + } + + var filterSelects = {} + function toggle() { + // selection options + + // selection.classed('d-flex flex-row', true) + // var filterButton = utils.sel.safeSelect(selection, 'a', 'slider-buttons') + // var filterI = utils.sel.safeSelect(filterButton, 'i', 'fa fa-sliders') + + /*BUG: unexpected behavior. + - when using bootstrap-eque way for collapse, clicking button submits to + the same page in applications (but not in demo), so using anchor () + - when using anchor, cause page to jump to that location + - when using show, the first open works, but the close does not. + */ + // filterButton.attr('class', 'btn btn-secondary') + // .attr('data-toggle', 'collapse') + // .attr('href', '#'+utils.str.hypenate(namespace, 'data-toggle')) + // .attr('target', '_blank') + // .html(filterButton.html()+' Filters') + // .on('click', function(d, i){ + // d3.event.preventDefault() + // d3.event.stopPropagation() + // var dt = d3.select("#"+utils.str.hypenate(namespace, 'data-toggle')) + // dt.classed('show', !dt.classed('show')) + // dt.classed('d-inline-flex', dt.classed('show')) + // + // filterButton.classed('btn-primary', dt.classed('show')) + // filterButton.classed('btn-secondary', !dt.classed('show')) + // }) + // .classed('d-inline-flex', true) + // .style("margin-right", '10px') + + + + // var datatoggleCollapse = utils.sel.safeSelect(selection, 'div', utils.str.hypenate(namespace,'collapse')) + // .attr('id', utils.str.hypenate(namespace, 'data-toggle')) + // .classed('collapse', true) + + // var flexRow = utils.sel.safeSelect(datatoggleCollapse, 'div', 'd-inline-flex flex-row flex-wrap') + var flexRow = utils.sel.safeSelect(selection, 'div', 'd-inline-flex flex-row flex-wrap') + + var dataopts = flexRow.selectAll('div.'+utils.str.hypenate(namespace,'select-filter')) + // remove excess + dataopts.exit().remove() + // bind data + dataopts = dataopts.data(d3.keys(data)) + //enter + var doEnter = dataopts.enter().append('div') + .attr('class', 'select-filter') + + dataopts = dataopts.merge(doEnter).style('margin-right', "10px") + + dataopts.each(function(d, i){ + var t = d3.select(this) + var sf = selectFilter(t) + .data(data[d]) + .namespace(utils.str.hypenate(namespace, d)) + .selectionName(d) + sf() + filterSelects[d] = sf + }) + + + selection.selectAll('select') + .on('change', function(){updateFunction()}) + // bind update function + // d3.selectAll + return toggle + } + + + function onlyOne() { + // d3.event.preventDefault() + // d3.event.stopPropagation() + var dataopts = selection.selectAll('div.data-option') + currentKey = dataopts.select(':checked').datum() + updateFunction() + + } + + function axisSelectFilter(selection, axis="x-axis", axisData) { + if (axisData == undefined) { + axisData = { + 'linear': d3.scaleLinear(), + 'log': d3.scaleLog(), + 'pow': d3.scalePow(), + 'sqrt': d3.scaleSqrt() + } + } + + var sf = selectFilter(selection) + .data(axisData) + .namespace(axis) + .selectionName(axis+' scale') + .defaultValue(d3.scaleLinear()) + + + sf() + + } + + return toggle +} diff --git a/src/modules/aux/index.js b/src/modules/aux/index.js index c52b890..b59c312 100644 --- a/src/modules/aux/index.js +++ b/src/modules/aux/index.js @@ -1,9 +1,13 @@ import lasso from './lasso' +import lassoWidget from './lasso-widget' import multiPlotZoom from './multi-plot-zoom' import plotZoom from './plot-zoom' +import datatoggle from './data-toggle' +import selectFilter from './select-filter' + let aux = { - lasso, plotZoom, multiPlotZoom + lasso, plotZoom, multiPlotZoom, datatoggle, selectFilter, lassoWidget } -export {lasso, plotZoom, multiPlotZoom} +export {lasso, plotZoom, multiPlotZoom, datatoggle, selectFilter, lassoWidget} export default aux diff --git a/src/modules/aux/lasso-widget.js b/src/modules/aux/lasso-widget.js new file mode 100644 index 0000000..c90eaca --- /dev/null +++ b/src/modules/aux/lasso-widget.js @@ -0,0 +1,604 @@ +import utils from '../utils/index.js' +import groupingSpacer from '../grouping-spacer'; +import CF from '../color-function'; +import TTip from '../tooltip'; +import lasso from './lasso'; +import '../d3-prototypes'; + +export default function lassoWidget( selection ) { + var + namespace = 'd3sm-lasso', + selection = selection, + svg, + chartContainer, + objectContainer, + objectClass, + lassoLine = d3.line() + .x(function(d, i){return d[0]}) + .y(function(d, i){return d[1]}) + .curve(d3.curveLinearClosed), + path, + colorFunction = CF(), + maxNumberOfGroups, + dataExtractor, + xScale, + yScale + + function onError(msg, style){ + console.log(msg, style) + } + // setup the container + setupDataselectContainer() + + lassoWidget.objectClass = function(_){return arguments.length ? (objectClass='.'+_.replace('.',''), lassoWidget) : objectClass} + lassoWidget.svg = function(_){return arguments.length ? (svg=_, lassoWidget) : svg} + lassoWidget.submit = function(_){return arguments.length ? (submit=_, lassoWidget) : submit} + lassoWidget.maxNumberOfGroups = function(_){return arguments.length ? (maxNumberOfGroups=_, lassoWidget) : maxNumberOfGroups} + lassoWidget.onError = function(_){return arguments.length ? (onError=_, lassoWidget) : onError} + lassoWidget.objectContainer = function(_){return arguments.length ? (objectContainer=_, lassoWidget) : objectContainer} + lassoWidget.chartContainer = function(_){return arguments.length ? (chartContainer=_, lassoWidget) : chartContainer} + lassoWidget.dataExtractor = function(_){return arguments.length ? (dataExtractor=_, lassoWidget) : dataExtractor} + lassoWidget.xScale = function(_){return arguments.length ? (xScale=_, lassoWidget) : xScale} + lassoWidget.yScale = function(_){return arguments.length ? (yScale=_, lassoWidget) : yScale} + + function styles() { + var s = d3.select("html").select("style."+namespace+'lasso-widget') + if (s.empty()) { + d3.select("html").append("style") + .classed(namespace+'lasso-widget', true) + .html( + "."+utils.str.hypenate(namespace, "data-table") + "{\ + height:100px;\ + overflow:auto;\ + }" + ) + } + } + + + function lassoWidget() { + styles() + + } + + function submit(data) { + console.log(data) + } + + function plusTab(tabList) { + var tab = utils.sel.safeSelect(tabList, 'li', utils.str.hypenate(namespace, 'plus-tab')) + .classed('ml-auto', true) + .classed('nav-item', true) + .on('click', makeNewGroup), + + anchor = utils.sel.safeSelect(tab, 'a', 'nav-link'), + icon = utils.sel.safeSelect(anchor, 'i', 'fa') + .classed('fa-plus fa-2x text-success', true) + } + + function sendTab(tabList) { + var tab = utils.sel.safeSelect(tabList, 'li', utils.str.hypenate(namespace, 'send-tab')) + .classed('ml-auto', true) + .classed('nav-item', true) + .on('click', clickSend) + , + + anchor = utils.sel.safeSelect(tab, 'a', 'nav-link'), + icon = utils.sel.safeSelect(anchor, 'i', 'fa') + .classed('fa-paper-plane-o fa-2x text-primary', true) + } + + function clickSend() { + var data = gatherDataLists() + submit(data) + } + + function closeTab(tabList) { + var tab = utils.sel.safeSelect(tabList, 'li', utils.str.hypenate(namespace, 'close-tab')) + .classed('ml-auto', true) + .classed('nav-item', true) + .on('click', function(){ + var m = d3.mouse(d3.select("html").node()) + selection.select('div.card').style("position", 'relative') + .transition().duration(1000) + .ease(d3.easeBack) + .style('left', window.outerWidth +'px') + .remove() + }) + , + + anchor = utils.sel.safeSelect(tab, 'a', 'nav-link'), + icon = utils.sel.safeSelect(anchor, 'i', 'fa') + .classed('fa-window-close-o fa-2x text-danger', true) + } + + + + function setupDataselectContainer() { + var + card = utils.sel.safeSelect(selection, 'div', 'card'), + cardHeader = utils.sel.safeSelect(card, 'div', 'card-header'), + + tabList = utils.sel.safeSelect(cardHeader, 'ul', 'nav') + .classed('nav-tabs card-header-tabs', true) + .classed(utils.str.hypenate(namespace, 'tab-list'), true) + .attr('role', 'tablist'), + + cardBody = utils.sel.safeSelect(card, 'div', 'card-body'), + tabContent = utils.sel.safeSelect(cardBody, 'div', 'tab-content') + .classed(utils.str.hypenate(namespace, 'tab-content'), true), + + // leftTabs = utils.sel.safeSelect(tabList, 'div', 'nav mr-auto left-aligned-tabs') + rightTabs = utils.sel.safeSelect(tabList, 'div', 'right-aligned-tabs').classed('nav ml-auto ', true) + + plusTab(rightTabs) + sendTab(rightTabs) + closeTab(rightTabs) + var + defaultTab = utils.sel.safeSelect(tabContent, 'div', 'tab-pane') + .classed(utils.str.hypenate(namespace,'default-tab'), true) + .classed("active", true) + .classed('text-left', true), + + p = utils.sel.safeSelect(defaultTab, 'div') + .html( + "Click to add a new group.
"+ + "Click to submit for re-analysis.
"+ + "Click to close the dataselect widget." + ) + // .text('Click the green plus to add a new group.') + } + + + function makeRemoveButton(paneBtnList) { + var + btn = utils.sel.safeSelect(paneBtnList, 'button', 'remove-btn') + .classed('btn btn-danger', true) + .on('click', removeBtnClickToRemove), + i = utils.sel.safeSelect(btn, 'i', 'fa').classed('fa-trash-o', true), + s = utils.sel.safeSelect(btn, 'span').text('Remove group') + return btn + } + + function removeBtnClickToRemove(d, i) { + var + tab = selection.select('#'+utils.str.hypenate(namespace,'tab',d)), + pane = selection.select('#'+utils.str.hypenate(namespace,'tab','pane',d)) + + tab.remove() + pane.remove() + + showRemainingPanes() + updateTabAndPaneNumbers() + } + + + function togglePaneById(id) { + var panes = selection.select('.'+utils.str.hypenate(namespace, 'tab-content')) + panes.selectAll('div.tab-pane') + .each(function(d, i){ + d3.select(this).classed('active show', d3.select(this).attr('id') == id) + }) + } + + function insertTab(tabs, n) { + + // var li = tabs.insert("li", ':nth-child('+(n)+')') + var li = tabs.insert("li", '.right-aligned-tabs') + .classed('nav-item', true) + .classed('alert', true) + .classed('alert-secondary', true) + .classed('alert-dismissable', true) + .classed('fade', true) + .classed('show', true) + // .classed('mr-auto', true) + .attr("role", 'alert') + .attr("id", utils.str.hypenate(namespace,'tab',n)) + .classed(utils.str.hypenate(namespace,'tab'), true) + + var a = utils.sel.safeSelect(li, 'a') + .attr('data-toggle', 'tab') + .text('Group ' + n) + .attr('href', '#'+utils.str.hypenate(namespace,'tab','pane',n)) + .on('dblclick', function(d, i){ + var t = d3.select(this) + t.attr('contenteditable', true) + d3.select(t.node().parentNode) + .classed('alert-secondary', false) + .classed('alert-warning', true) + + // d3.select("html").on("click", dispatchBlur) + // + // function dispatchBlur(d, i) { + // if (d3.event.target != d3.select(this)) { + // d3.select(this).dispatch("blur") + // } + // } + + + }) + .on('blur', function(d, i){ + + var t = d3.select(this) + t.attr('contenteditable', false) + d3.select(t.node().parentNode) + .classed('alert-secondary', true) + .classed('alert-warning', false) + + }) + .on('input', function(d, i){ + var t = d3.select(this) + var txt = t.text() + d3.select(t.attr('href')).select("p.lead").text(txt) + }) + + + return li + } + + function makeTabPane(panes, n) { + var pane = panes.append('div') + .datum(n) + .attr('role', 'tabpanel') + .classed('tab-pane', true) + .classed('text-left', true) + .classed(utils.str.hypenate(namespace,'tab','pane'), true) + .attr('id', utils.str.hypenate(namespace,'tab','pane',n)) + return pane + } + + function populatePane(pane, n) { + var + lead = utils.sel.safeSelect(pane, 'p', 'lead').text('Group '+n), + + tableContainer = utils.sel.safeSelect(pane, 'div', 'table-responsive') + .attr("class", utils.str.hypenate(namespace, "data-table")), + + table = utils.sel.safeSelect(tableContainer, "table", "table") + .classed("table-sm", true).classed("table-hover", true), + + caption = utils.sel.safeSelect(table, "caption", "caption").html("List of selected"), + + btns = utils.sel.safeSelect(pane, 'div', 'text-right'), + + cN = n % colorFunction.colors().length + if (n % 2 != 0) { cN = (colorFunction.colors().length - 1) - cN } + + + + + + var lassoBtn = makeLassoButton(btns) + var currentLasso = makeLassoFunction(n, colorFunction.colors()[cN]) + var clearBtn = makeClearButton(btns) + + bindButtons(lassoBtn, clearBtn, currentLasso, table) + + var removeBtn = makeRemoveButton(btns) + + lead.on("mouseover", function(){ + currentLasso.draw() + }) + lead.on("mouseout", function(){ + currentLasso.remove() + }) + + } + + function makeLassoButton(btns) { + var btn = utils.sel.safeSelect(btns, 'button', 'lasso-btn') + .classed('btn btn-info', true) + .classed(namespace, true) + // .datum([lasso]) + var i = utils.sel.safeSelect(btn, 'i', 'fa').classed('fa-hand-pointer-o', true) + var s = utils.sel.safeSelect(btn, 'span').text('Lasso select') + return btn + + } + + function makeClearButton(btns) { + var btn = utils.sel.safeSelect(btns, 'button', 'clear-btn') + .classed('btn btn-light', true) + .classed(namespace, true) + var i = utils.sel.safeSelect(btn, 'i', 'fa').classed('fa-eraser', true) + var s = utils.sel.safeSelect(btn, 'span').text('Clear selection') + return btn + } + + function makeLassoFunction(instance, color) { + var currentLasso = lasso() + .namespace(namespace) + .svg(svg) + .objectClass(objectClass) + .chartContainer(chartContainer) + .objectContainer(objectContainer) + .instance(instance) + .color(color) + .yScale(yScale) + .xScale(xScale) + + return currentLasso + } + + function bindButtons(lassoBtn, clearBtn, currentLasso, table) { + currentLasso.eventCatcher(lassoBtn) + lassoBtn.node().addEventListener("click", lassoBtnToggle) + + lassoBtn.datum([currentLasso]) + .on(utils.str.hypenate(namespace,'render'), function(){ + d3.select(this).datum()[0].draw() + }) + .on(utils.str.hypenate(namespace,"drag"), function(las){ + var nodes = chartContainer.selectAll(".in-lasso-"+las[0].instance()).nodes() + var tableData = nodes.map(function(d, i){ + var extracted = dataExtractor(d3.select(d).datum()) + extracted["__node"] = d + return extracted + }) + + + makeTable(table, tableData, currentLasso) + }) + + clearBtn.on('click', function(){ + currentLasso.allPoints([]) + currentLasso.currentPoints([]) + lassoBtn.dispatch(utils.str.hypenate(namespace,"drag")) + + }) + + } + + function lassoBtnToggle() { + var that = d3.select(this) + var las = that.datum()[0] + + las.toggle() + var activeQ = las.activeQ() + + if (las.activeQ()) { + that.classed('btn-info', !activeQ) + that.classed('btn-warning', activeQ) + that.select("span").text("Lasso select (active)") + selection.selectAll("."+namespace+".lasso-btn").dispatch(utils.str.hypenate(namespace,'render')) + d3.select("html").node().addEventListener('mousedown', monitorLassoButtonState) + } else { + that.classed('btn-info', !activeQ) + that.classed('btn-warning', activeQ) + that.select("span").text("Lasso select") + d3.select("html").node().removeEventListener('mousedown', monitorLassoButtonState) + } + + function monitorLassoButtonState(event) { + /* + activeLasso stops event stopPropagation, so this event will not register in + that case. Thus we only need to ensure that the usre is clicking on the lasso button + otherwise, de-spawn lasso. + */ + if ( + event.target != that.node() && + event.target != that.select("span").node() && + event.target != that.select("i").node() + ) { + // event.preventDefault() + // event.stopPropagation() + las.toggle(false) + that.classed('btn-info', true) + that.classed('btn-warning', false) + that.select("span").text("Lasso select") + // console.log(that, that.node()) + // that.dispatch("focusout") + } + } + + } + + + function updateTableHeaderColumns(headR, tableData) { + var headerKeys = d3.keys(tableData[0]).filter(k=>k!="__node") + if (headerKeys.length > 0) { + // headerKeys = ["remove"].concat(headerKeys) + headerKeys.push("remove") + } + + var headerCols = headR.selectAll("th") + + headerCols = headerCols.data(headerKeys) + headerCols.exit().remove() + headerCols = headerCols.merge(headerCols.enter().append("th").attr("scope","col")) + .text(function(d, i){return d}) + + return headerKeys + } + + function updateTableRows(body, tableData) { + var bodyRows = body.selectAll("tr") + + bodyRows = bodyRows.data(tableData) + bodyRows.exit().remove() + bodyRows = bodyRows.merge(bodyRows.enter().append("tr")) + + return bodyRows + } + + function updateTableRowColumns(cols, headerKeys, rowData, tableData, lasso, table) { + cols = cols.data(headerKeys) + cols.exit().remove() + cols = cols.merge(cols.enter().append("td")) + + cols.html(function(d, i){ + if (d != "remove") { return rowData[d] } + return "" + }).classed("text-left", function(d, i){ + if (d != "remove") { return false } + return true + }) + // .attr("scope",function(d, i){ + // if (d != "remove") { return false } + // return true + // }) + + cols.select("i.fa-close").on("click", function(d, i){ + var node = rowData["__node"], + n = d3.select(node) + n.classed("in-lasso", false) + n.classed("in-lasso-"+lasso.instance(), false) + + tableData.map(function(dd, j){ + if (dd["__node"] == node) { + tableData.splice(j, 1) + makeTable(table, tableData, lasso) + } + }) + + }) + } + + function makeTable(table, tableData, lasso) { + var + head = utils.sel.safeSelect(table, "thead"), + headR = utils.sel.safeSelect(head, "tr"), + body = utils.sel.safeSelect(table, "tbody"), + + headerKeys = updateTableHeaderColumns(headR, tableData), + + bodyRows = updateTableRows(body, tableData) + + bodyRows.each(function(rowData, i){ + var t = d3.select(this) + var cols = t.selectAll("td") + + updateTableRowColumns(cols, headerKeys, rowData, tableData, lasso, table) + + t.on("mouseover", function(d, i) { + lasso.applyObjectAttributes(d3.select(d["__node"]),true) + }) + .on("mouseout", function(d, i) { + lasso.applyObjectAttributes(d3.select(d["__node"]),false) + }) + }) + + + } + + + + + + + + function makeNewGroup(d, i) { + + var tabs = selection.select('.'+utils.str.hypenate(namespace, 'tab-list')), + panes = selection.select('.'+utils.str.hypenate(namespace, 'tab-content')), + n = newGroupNumber() + + if (tabs.selectAll('.'+utils.str.hypenate(namespace,'tab')).size() == maxNumberOfGroups) { + onError('only '+maxNumberOfGroups+' allowed.', 'warning') + return + } + + var + tab = insertTab(tabs, n), + pane = makeTabPane(panes, n) + + populatePane(pane, n) + togglePaneById(pane.attr('id')) + + } + + function newGroupNumber() { + var tabs = selection.select('.'+utils.str.hypenate(namespace, 'tab-list'))//, + var + n = tabs.selectAll('li').size() - 3, // minus 1 for plus tab + s = 'Group ' + n, + gs = [] + tabs.each(function(d, i){ return gs.push(d3.select(this).select("a").text()) }) + while (gs.includes(s)) { n+=1; s = 'Group ' + n; } + return n + } + + function updateTabAndPaneNumbers() { + var + tabs = selection.select('.'+utils.str.hypenate(namespace, 'tab-list')), + panes = selection.select('.'+utils.str.hypenate(namespace, 'tab-content')) + + tabs = tabs.selectAll('.'+utils.str.hypenate(namespace,'tab')) + panes = panes.selectAll('.'+utils.str.hypenate(namespace,'tab', 'pane')) + + tabs.each(function(d, i){ + d3.select(this).datum(i) + .attr("id", utils.str.hypenate(namespace,'tab',i)) + .select('a') + .attr('href', '#'+utils.str.hypenate(namespace,'tab','pane',i)) + .text(function(dd, ii){ + var curText = d3.select(this).text(); + if (curText.split(' ')[0] == 'Group') { + return 'Group ' + i + } + return curText + }) + }) + + panes.each(function(d, i){ + d3.select(this).datum(i) + .attr('id', utils.str.hypenate(namespace,'tab','pane',i)) + utils.sel.safeSelect(d3.select(this), 'p', 'lead') + .text(function(dd, ii){ + var curText = d3.select(this).text(); + if (curText.split(' ')[0] == 'Group') { + return 'Group ' + i + } + return curText + }) + utils.sel.safeSelect(d3.select(this), 'button', 'remove-btn') + .on('click', removeBtnClickToRemove) + }) + + } + + function gatherDataLists() { + var tabs = selection.select('.'+utils.str.hypenate(namespace, 'tab-list')) + var panes = selection.select('.'+utils.str.hypenate(namespace, 'tab-content')) + var tables = panes.selectAll('.'+utils.str.hypenate(namespace, "data-table")) + var data = {} + + var textGroups = tabs.selectAll('li.'+utils.str.hypenate(namespace,'tab') + ' > a') + .nodes().map(function(d, i){return d3.select(d).text()}) + + textGroups.map(function(e, i){ + data[e] = [] + }) + + + tables.each(function(d, i){ + var table = d3.select(this).select("tbody") + data[textGroups[i]] = table.selectAll('tr').data() + }) + + return data + } + + function showRemainingPanes() { + var + tabs = selection.select('.'+utils.str.hypenate(namespace, 'tab-list')), + panes = selection.select('.'+utils.str.hypenate(namespace, 'tab-content')), + remainingTabs = tabs.selectAll('.'+utils.str.hypenate(namespace,'tab')) + + if (remainingTabs.size() == 0) { + panes.select('.'+utils.str.hypenate(namespace,'default-tab')) + .classed("active", true) + .classed('text-left', true) + } + else { + var lastTab = remainingTabs.nodes()[remainingTabs.size()-1], + lastPaneId = d3.select(lastTab).select('a').attr('href') + panes.select(lastPaneId) + .classed("active", true) + .classed('text-left', true) + } + } + + + return lassoWidget +} diff --git a/src/modules/aux/lasso.js b/src/modules/aux/lasso.js index d3bbb24..78e3049 100644 --- a/src/modules/aux/lasso.js +++ b/src/modules/aux/lasso.js @@ -4,13 +4,13 @@ import CF from '../color-function'; import TTip from '../tooltip'; import '../d3-prototypes'; -// function getTranslation(selection){ -// var transform = selection.attr('transform') -// var [junk, xy] =transform.split('translate(') -// var [x, y] = xy.split(',') -// y, junk = y.split(')') -// return [parseFloat(x), parseFloat(y)] -// } +function getTranslation(selection){ + var transform = selection.attr('transform') + var [junk, xy] =transform.split('translate(') + var [x, y] = xy.split(',') + y, junk = y.split(')') + return [parseFloat(x), parseFloat(y)] +} export default function lasso( selection ) { var @@ -112,8 +112,8 @@ export default function lasso( selection ) { function toggle(state) { // use optional param to set state, otherwise toggle state activeQ = (state!=undefined) ? state : !activeQ - chartOffset = utils.sel.getTranslation(chartContainer) - objectsOffset = utils.sel.getTranslation(objectContainer) + chartOffset = getTranslation(chartContainer) //utils.sel.getTranslation(chartContainer) + objectsOffset = getTranslation(objectContainer) //utils.sel.getTranslation(objectContainer) if (activeQ) { svg.node().addEventListener('mousedown', render, true) @@ -125,8 +125,8 @@ export default function lasso( selection ) { } function draw() { - chartOffset = utils.sel.getTranslation(chartContainer) - objectsOffset = utils.sel.getTranslation(objectContainer) + chartOffset = getTranslation(chartContainer) //utils.sel.getTranslation(chartContainer) + objectsOffset = getTranslation(objectContainer) //utils.sel.getTranslation(objectContainer) var container = utils.sel.safeSelect(objectContainer, 'g', 'lasso-container') var paths = container.selectAll('path[instance="'+instance+'"]') diff --git a/src/modules/aux/select-filter.js b/src/modules/aux/select-filter.js new file mode 100644 index 0000000..288eb41 --- /dev/null +++ b/src/modules/aux/select-filter.js @@ -0,0 +1,110 @@ +import utils from '../utils/index.js' + +export default function selectFilter(selection) { + + var + data, + namespace = 'd3sm-select-filter', + selectionName = 'Select options:', + defaultValue = undefined + + + + + var lastValue = undefined + + selectFilter.data = function(_) { return arguments.length ? (data = _, selectFilter) : data} + selectFilter.namespace = function(_) { return arguments.length ? (namespace = _, selectFilter) : namespace} + selectFilter.selectionName = function(_) { return arguments.length ? (selectionName = _, selectFilter) : selectionName} + selectFilter.defaultValue = function(_) { return arguments.length ? (defaultValue = _, selectFilter) : defaultValue} + selectFilter.currentOption = currentOption + + function selectFilter() { + var + container = utils.sel.safeSelect(selection, 'div', 'input-group').classed(utils.str.hypenate(namespace,'container'),true), + + selectPrepend = utils.sel.safeSelect(container, 'div', 'select-prepend').classed('input-group-prepend', true), + selectPrependSpan = utils.sel.safeSelect(selectPrepend, 'span', 'input-group-text').text(selectionName), + + select = utils.sel.safeSelect(container, 'select', 'custom-select').classed(utils.str.hypenate(namespace,'select'),true), + + selectAppend = utils.sel.safeSelect(container, 'div', 'select-append').classed('input-group-prepend', true), + selectAppendButton = utils.sel.safeSelect(selectAppend, 'a', 'filter-button').classed('btn btn-outline-secondary', true), + filterButtonIcon = utils.sel.safeSelect(selectAppendButton, 'i', 'fa fa-filter'), + + inputGroup = utils.sel.safeSelect(container, 'div', 'filter-input-group').classed('input-group',true).classed('d-none', true), + inputPrepend = utils.sel.safeSelect(inputGroup, 'div', 'input-group-prepend'), + inputPrependSpan = utils.sel.safeSelect(inputPrepend, 'span', 'input-group-text').classed('search-button', true), + inputPrependSpanIcon = utils.sel.safeSelect(inputPrependSpan,'i','fa fa-search'), + + input = utils.sel.safeSelect(inputGroup, 'input', 'form-control').attr('placeholder', 'all').attr('type', 'text'), + inputAppend = utils.sel.safeSelect(inputGroup, 'div', 'input-group-append'), + inputAppendButton = utils.sel.safeSelect(inputAppend, 'a', 'close-button').classed('btn btn-outline-secondary', true), + inputAppendButtonIcon = utils.sel.safeSelect(inputAppendButton, 'i', 'fa fa-close') + + + var keys = d3.keys(data), + options = select.selectAll('option') + + options = options.data(d3.keys(data)) + options = options.merge(options.enter().append('option')) + .attr('value', function(d, i){return d}) + .text(function(d, i){return d}) + + var + filterButton = selectAppendButton, + closeButton = inputAppendButton + + filterButton.on('click', function(d, i){ + var currentStyle = inputGroup.classed('d-none') + inputGroup.classed('d-none', !currentStyle) + }) + + closeButton.on('click', function(d, i){ + input.property('value', '').dispatch('input') + }) + + input.on('input', function(d, i){ + var + val = input.property('value'), + reg = new RegExp(val, 'gi'), + use + + if (val == '') {use = keys} + else { + use = [] + d3.keys(data).map(function(option, j){ + var match = option.match(reg) + if (match == null || match.join('') == '') {} + else { use.push(option) } + }) + } + + options = select.selectAll('option') + options = options.data(use) + options.exit().remove() + options = options.merge(options.enter().append('option')) + .attr('value', function(d, i){return d}) + .text(function(d, i){return d}) + + var current = currentOption() + if (lastValue != current) { + lastValue = current + select.dispatch('change') + } + }) + + + } + + function currentOption() { + var val = selection.select("select").property('value') + return val == undefined || val == '' + ? defaultValue == undefined + ? d3.keys(data)[0] + : defaultValue + : val + } + + return selectFilter +} diff --git a/src/modules/charts/box-whisker.js b/src/modules/charts/box-whisker.js index 55e0638..ee7121f 100644 --- a/src/modules/charts/box-whisker.js +++ b/src/modules/charts/box-whisker.js @@ -18,7 +18,7 @@ import TTip from '../tooltip'; * @namespace boxwhisker * @returns {function} boxwhisker */ -export function boxwhisker( selection ) { +export default function boxwhisker( selection ) { var /** * Data to plot. Assumed to be a object, where each key corresponds to a box diff --git a/src/modules/charts/index.js b/src/modules/charts/index.js index 05d8b08..7f9cc56 100644 --- a/src/modules/charts/index.js +++ b/src/modules/charts/index.js @@ -1,14 +1,15 @@ import scatter from './scatter' import bar from './bar' import bubble from './bubble-heatmap' +import boxwhisker from './box-whisker' import heatmap from './heatmap' import violin from './violin' import {neededViolinValues} from './violin' import upset from './upset' let charts = { - scatter, bar, bubble, heatmap, violin, neededViolinValues, upset + scatter, bar, bubble, boxwhisker, heatmap, violin, neededViolinValues, upset } -export {scatter, bar, bubble, heatmap, violin, neededViolinValues, upset} +export {scatter, bar, bubble, boxwhisker, heatmap, violin, neededViolinValues, upset} export default charts diff --git a/src/modules/charts/violin.js b/src/modules/charts/violin.js index 564e493..ca6651b 100644 --- a/src/modules/charts/violin.js +++ b/src/modules/charts/violin.js @@ -1,3 +1,4 @@ +import utils from '../utils/index.js'; import groupingSpacer from '../grouping-spacer'; import CF from '../color-function'; import TTip from '../tooltip'; @@ -682,12 +683,16 @@ export default function violin( selection ) { // augment valus - violinKeys.map(function(vk, i){ calcValues(vk, data) }) + violinValues = {} + violinKeys.map(function(vk, i){ + let v = calcValues(vk, data) + violinValues[vk] = v + }) var numberOfObjects = violinKeys.length - var min = [].concat(...violinKeys.map(function(k, i){return data[k].utils.math.quartiles[quartileKeys[0]]})) - var max = [].concat(...violinKeys.map(function(k, i){return data[k].utils.math.quartiles[quartileKeys[quartileKeys.length - 1]]})) + var min = [].concat(...violinKeys.map(function(k, i){return data[k].quartiles[quartileKeys[0]]})) + var max = [].concat(...violinKeys.map(function(k, i){return data[k].quartiles[quartileKeys[quartileKeys.length - 1]]})) var extent = [Math.min(...min) - domainPadding, Math.max(...max) + domainPadding] // console.log(extent, violinValues, ordered) @@ -754,9 +759,9 @@ export default function violin( selection ) { quarts = utils.sel.safeSelect(t, 'g', 'quarts'), lq3 = utils.sel.safeSelect(quarts, 'line', 'q3'), lq1 = utils.sel.safeSelect(quarts, 'line', 'q1'), - q3 = currentData.utils.math.quartiles[quartileKeys[3]], - q2 = currentData.utils.math.quartiles[quartileKeys[2]], - q1 = currentData.utils.math.quartiles[quartileKeys[1]] + q3 = currentData.quartiles[quartileKeys[3]], + q2 = currentData.quartiles[quartileKeys[2]], + q1 = currentData.quartiles[quartileKeys[1]] t.attr('transform', horizontalQ ? 'translate('+objectSize / 2+',0)' : 'translate(0,'+objectSize / 2+')' ) // draw curve @@ -873,11 +878,11 @@ export default function violin( selection ) { tooltip() if (tooltip.values() == undefined) { tooltip.values([ - function(currentData, tooltipKey){ return currentData['utils.math.quartiles'][tooltipKey] }, - function(currentData, tooltipKey){ return currentData['utils.math.quartiles'][tooltipKey] }, - function(currentData, tooltipKey){ return currentData['utils.math.quartiles'][tooltipKey] }, - function(currentData, tooltipKey){ return currentData['utils.math.quartiles'][tooltipKey] }, - function(currentData, tooltipKey){ return currentData['utils.math.quartiles'][tooltipKey] } + function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] }, + function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] }, + function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] }, + function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] }, + function(currentData, tooltipKey){ return currentData['quartiles'][tooltipKey] } ]) } @@ -1036,9 +1041,10 @@ export function neededViolinValues() { violinData.binned = binned; violinData.frequencies = frequencies violinData.contour = violinContourPoints - violinData.utils.math.quartiles = pointQuartiles + violinData.quartiles = pointQuartiles violinData.pointKeys = violinPointsKeys violinData.pointValues = violinPointsValues + return violinData } return calculateViolinValues diff --git a/src/modules/d3-prototypes.js b/src/modules/d3-prototypes.js index 3033b24..13c50e7 100644 --- a/src/modules/d3-prototypes.js +++ b/src/modules/d3-prototypes.js @@ -40,7 +40,7 @@ d3.mouse.absolute = function() { d3.selection.prototype.absolutePosition = function() { var element = this.node(); var elementPosition = element.getBoundingClientRect(); - var containerSVG = getContainingSVG(element) + var containerSVG = utils.sel.getContainingSVG(element) var svgPosition = containerSVG.getBoundingClientRect(); return { diff --git a/src/modules/utils/math.js b/src/modules/utils/math.js index eac6234..824475f 100644 --- a/src/modules/utils/math.js +++ b/src/modules/utils/math.js @@ -1,3 +1,4 @@ +import {total} from './array.js' /** * Rounds decimals of number to precision * @param {number} number diff --git a/src/modules/utils/misc.js b/src/modules/utils/misc.js index ac8344f..cbc3dad 100644 --- a/src/modules/utils/misc.js +++ b/src/modules/utils/misc.js @@ -1,4 +1,5 @@ - +import {safeSelect} from './selections.js' +import {hypenate} from './strings.js' export function resizeDebounce(f, wait) { var resize = debounce(function(){f()},wait) window.addEventListener('resize', resize) @@ -96,22 +97,22 @@ export function setupStandardChartContainers( - container = d3sm.safeSelect(selection, 'svg', namespace) + container = safeSelect(selection, 'svg', namespace) .style('width', svgSpace.w+'px') .style('height', svgSpace.h+'px') - var axes = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'axes')) + var axes = safeSelect(container, 'g', hypenate(namespace, 'axes')) - var leg = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'legend')) + var leg = safeSelect(container, 'g', hypenate(namespace, 'legend')) .attr('transform', "translate("+legRect.x+","+legRect.y+")") - var plot = d3sm.safeSelect(container, 'g', d3sm.hypenate(namespace, 'plot')) + var plot = safeSelect(container, 'g', hypenate(namespace, 'plot')) .attr('transform', "translate("+plotRect.x+","+plotRect.y+")") - var xAxis = d3sm.safeSelect(axes, 'g', d3sm.hypenate(namespace, 'x-axis')) + var xAxis = safeSelect(axes, 'g', hypenate(namespace, 'x-axis')) .attr('transform', "translate("+xAxisRect.x+","+xAxisRect.y+")") - var yAxis = d3sm.safeSelect(axes, 'g', d3sm.hypenate(namespace, 'y-axis')) + var yAxis = safeSelect(axes, 'g', hypenate(namespace, 'y-axis')) .attr('transform', "translate("+yAxisRect.x+","+yAxisRect.y+")") return { -- GitLab