timeline.js/timeline.min.js
Phyks 8809cef704 * Better grid : autoscale + good positionning for y axis
* Value available in legend using %v
2014-03-29 17:28:20 +01:00

1 line
11 KiB
JavaScript

var SVG={};SVG.ns="http://www.w3.org/2000/svg";SVG.xlinkns="http://www.w3.org/1999/xlink";SVG.marginBottom=10;SVG.marginTop=15;SVG.marginLeft=10;SVG.marginRight=10;SVG.rounded=false;SVG.x_axis=false;SVG.parent_holder=false;SVG.holder=false;SVG.g=false;SVG.axis=false;SVG.raw_points=[];SVG.labels=[];SVG.init=function(f,v,g,e,u,j){if(!document.implementation.hasFeature("http://www.w3.org/TR/SVG11/feature#Image","1.1")){alert("Your browser does not support embedded SVG.")}SVG.parent_holder=document.getElementById(f);var s=document.createElementNS(SVG.ns,"svg:svg");s.setAttribute("width",g);s.setAttribute("height",v);s.setAttributeNS("http://www.w3.org/2000/xmlns/","xmlns:xlink",SVG.xlinkns);SVG.parent_holder.appendChild(s);SVG.holder=SVG.parent_holder.querySelector("svg");var o=document.createElementNS(SVG.ns,"defs");SVG.holder.appendChild(o);if(e==="small"||e==="both"){var t=document.createElementNS(SVG.ns,"pattern");t.setAttribute("id","smallGrid");t.setAttribute("width",8);t.setAttribute("height",8);t.setAttribute("patternUnits","userSpaceOnUse");var l=document.createElementNS(SVG.ns,"path");l.setAttribute("d","M 8 0 L 0 0 0 8");l.setAttribute("fill","none");l.setAttribute("stroke","gray");l.setAttribute("stroke-width","0.5");t.appendChild(l);o.appendChild(t)}if(e==="big"||e==="both"){var k=document.createElementNS(SVG.ns,"pattern");k.setAttribute("id","grid");k.setAttribute("width",80);k.setAttribute("height",80);k.setAttribute("patternUnits","userSpaceOnUse");if(e==="both"){var h=document.createElementNS(SVG.ns,"rect");h.setAttribute("width",80);h.setAttribute("height",80);h.setAttribute("fill","url(#smallGrid)");k.appendChild(h)}var q=document.createElementNS(SVG.ns,"path");q.setAttribute("d","M 80 0 L 0 0 0 80");q.setAttribute("fill","none");q.setAttribute("stroke","gray");q.setAttribute("stroke-width","1");k.appendChild(q);o.appendChild(k)}SVG.grid=e;var p=document.createElementNS(SVG.ns,"marker");p.setAttribute("id","markerArrow");p.setAttribute("markerWidth",13);p.setAttribute("markerHeight",13);p.setAttribute("refX",2);p.setAttribute("refY",6);p.setAttribute("orient","auto");var d=document.createElementNS(SVG.ns,"path");d.setAttribute("d","M2,2 L2,11 L10,6 L2,2");d.setAttribute("fill","gray");p.appendChild(d);o.appendChild(p);SVG.g=document.createElementNS(SVG.ns,"g");SVG.g.setAttribute("transform","translate(0, "+SVG.parent_holder.offsetHeight+") scale(1, -1)");SVG.holder.appendChild(SVG.g);if(u===true){SVG.axis=document.createElementNS(SVG.ns,"line");SVG.axis.setAttribute("x1",SVG.marginLeft);SVG.axis.setAttribute("x2",SVG.parent_holder.offsetWidth-13-SVG.marginRight);SVG.axis.setAttribute("stroke","gray");SVG.axis.setAttribute("stroke-width",3);SVG.axis.setAttribute("marker-end",'url("#markerArrow")');SVG.g.appendChild(SVG.axis)}if(SVG.grid!=="none"){var e=document.createElementNS(SVG.ns,"rect");e.setAttribute("width","100%");e.setAttribute("height","100%");if(SVG.grid==="big"||SVG.grid==="both"){e.setAttribute("fill","url(#grid)")}else{e.setAttribute("fill","url(#smallGrid)")}SVG.g.appendChild(e)}SVG.rounded=j;SVG.x_axis=u};SVG.newCoordinates=function(h,f,d,g,j){var e=(j-g)/(d-f);return e*h-e*f+g};SVG.scale=function(k){var h=false,g=0;var e=false,d=false;var p=0,o=0;var f=false,l=false,q=false;for(point=0;point<k.length;point++){p=k[point][0];o=k[point][1];if(p<h||h===false){h=p}if(p>e||e===false){e=p}if(o<g){g=o}if(o>d||d===false){d=o}}o=SVG.newCoordinates(g+Math.pow(10,Math.floor(Math.log(d-g)/Math.log(10))),g,d,2*SVG.marginBottom,SVG.parent_holder.offsetHeight-SVG.marginTop)-SVG.newCoordinates(g,g,d,2*SVG.marginBottom,SVG.parent_holder.offsetHeight-SVG.marginTop);if(SVG.grid==="big"||SVG.grid==="both"){SVG.holder.getElementById("grid").setAttribute("width",o);SVG.holder.getElementById("grid").setAttribute("height",o);SVG.holder.getElementById("grid").setAttribute("y",SVG.newCoordinates(Math.floor(g/Math.pow(10,Math.floor(Math.log(d-g)/Math.log(10))))*Math.pow(10,Math.floor(Math.log(d-g)/Math.log(10))),g,d,2*SVG.marginBottom,SVG.parent_holder.offsetHeight-SVG.marginTop));SVG.holder.getElementById("grid").querySelector("path").setAttribute("d","M "+o+" 0 L 0 0 0 "+o);if(SVG.grid==="both"){SVG.holder.getElementById("grid").querySelector("rect").setAttribute("width",o);SVG.holder.getElementById("grid").querySelector("rect").setAttribute("height",o)}}if(SVG.grid==="small"||SVG.grid==="both"){o=o/10;SVG.holder.getElementById("smallGrid").setAttribute("width",o);SVG.holder.getElementById("smallGrid").setAttribute("height",o);if(SVG.grid==="small"){SVG.holder.getElementById("smallGrid").setAttribute("y",SVG.newCoordinates(Math.floor(g/Math.pow(10,Math.floor(Math.log(d-g)/Math.log(10))))*Math.pow(10,Math.floor(Math.log(d-g)/Math.log(10))),g,d,2*SVG.marginBottom,SVG.parent_holder.offsetHeight-SVG.marginTop))}SVG.holder.getElementById("smallGrid").querySelector("path").setAttribute("d","M "+o+" 0 L 0 0 0 "+o);SVG.holder.getElementById("smallGrid").querySelector("path").setAttribute("d","M "+o+" 0 L 0 0 0 "+o)}if(SVG.x_axis===true){o=SVG.newCoordinates(0,g,d,2*SVG.marginBottom,SVG.parent_holder.offsetHeight-SVG.marginTop);SVG.axis.setAttribute("y1",o);SVG.axis.setAttribute("y2",o)}var j=new Array();j.minX=h;j.minY=g;j.maxX=e;j.maxY=d;return j};SVG.addPoints=function(d){d.sort(function(f,e){if(f.x<e.x){return -1}else{if(f.x==e.x){return 0}else{return 1}}});for(point=0;point<d.length;point++){SVG.raw_points.push([d[point].x,d[point].y]);if(d[point].label!=="undefined"){SVG.labels.push(d[point].label)}else{SVG.labels.push("")}}};SVG.getControlPoints=function(d){p1=new Array();p2=new Array();n=d.length-1;a=new Array();b=new Array();c=new Array();r=new Array();a[0]=0;b[0]=2;c[0]=1;r[0]=d[0]+2*d[1];for(i=1;i<n-1;i++){a[i]=1;b[i]=4;c[i]=1;r[i]=4*d[i]+2*d[i+1]}a[n-1]=2;b[n-1]=7;c[n-1]=0;r[n-1]=8*d[n-1]+d[n];for(i=1;i<n;i++){m=a[i]/b[i-1];b[i]=b[i]-m*c[i-1];r[i]=r[i]-m*r[i-1]}p1[n-1]=r[n-1]/b[n-1];for(i=n-2;i>=0;--i){p1[i]=(r[i]-c[i]*p1[i+1])/b[i]}for(i=0;i<n-1;i++){p2[i]=2*d[i+1]-p1[i+1]}p2[n-1]=0.5*(d[n]+p1[n-1]);return{p1:p1,p2:p2}};SVG.draw=function(){var e=SVG.scale(SVG.raw_points);var u=new Array();var q=new Array();var k=false;var s=false;var w=false;var v=false;var t=false;var A="";for(w=0;w<SVG.raw_points.length;w++){u.push(SVG.newCoordinates(SVG.raw_points[w][0],e.minX,e.maxX,SVG.marginLeft,SVG.parent_holder.offsetWidth-SVG.marginRight));q.push(SVG.newCoordinates(SVG.raw_points[w][1],e.minY,e.maxY,2*SVG.marginBottom,SVG.parent_holder.offsetHeight-SVG.marginTop))}if(SVG.rounded===true){v=SVG.getControlPoints(u);t=SVG.getControlPoints(q);for(w=0;w<SVG.raw_points.length-1;w++){A+="C "+v.p1[w]+" "+t.p1[w]+" "+v.p2[w]+" "+t.p2[w]+" "+u[w+1]+" "+q[w+1]+" "}}else{for(w=1;w<SVG.raw_points.length;w++){A+="L "+u[w]+" "+q[w]+" "}}k=document.createElementNS(SVG.ns,"path");k.setAttribute("class","graph");k.setAttribute("fill","#3f72bf");k.setAttribute("opacity","0.25");k.setAttribute("stroke","none");k.setAttribute("d","M "+SVG.marginLeft+" "+SVG.marginBottom+" L "+u[0]+" "+q[0]+" "+A+" L "+(SVG.parent_holder.offsetWidth-SVG.marginRight)+" "+SVG.marginBottom+" Z");SVG.g.insertBefore(k,SVG.g.querySelectorAll(".over")[0]);k=document.createElementNS(SVG.ns,"path");k.setAttribute("class","line");k.setAttribute("stroke","#3f72bf");k.setAttribute("stroke-width",2);k.setAttribute("fill","none");k.setAttribute("d","M "+u[0]+" "+q[0]+" "+A);SVG.g.appendChild(k);for(w=0;w<SVG.raw_points.length;w++){k=document.createElementNS(SVG.ns,"circle");k.setAttribute("class","point");k.setAttribute("id","point_"+w);k.setAttribute("cx",u[w]);k.setAttribute("cy",q[w]);k.setAttribute("r",4);k.setAttribute("fill","#333");k.setAttribute("stroke","#3f72bf");k.setAttribute("stroke-width",2);SVG.g.appendChild(k);if(SVG.labels[w]!==""){var l=document.createElementNS(SVG.ns,"g");l.setAttribute("class","label");l.setAttribute("id","label_"+w);l.setAttribute("transform","translate(0, "+SVG.parent_holder.offsetHeight+") scale(1, -1)");SVG.g.appendChild(l);k=document.createElementNS(SVG.ns,"text");var z=SVG.labels[w].replace("</sup>","<sup>").split("<sup>");var j=0;var h=false;for(j=0;j<z.length;j++){z[j]=z[j].replace(/(<([^>]+)>)/ig,"").replace("%v",SVG.raw_points[w][1]);if(j%2==0){k.appendChild(document.createTextNode(z[j]))}else{h=document.createElementNS(SVG.ns,"tspan");h.setAttribute("dy","-5");h.appendChild(document.createTextNode(z[j]));k.appendChild(h)}}var A=document.createElementNS(SVG.ns,"path");A.setAttribute("stroke","black");A.setAttribute("stroke-width",2);A.setAttribute("fill","white");A.setAttribute("opacity",0.5);l.appendChild(A);l.appendChild(k);var f=u[w]-k.getBoundingClientRect().width/2;var p=SVG.parent_holder.offsetHeight-q[w]-20;var o=k.getBoundingClientRect().width;var d=k.getBoundingClientRect().height;if(u[w]+o/2>SVG.parent_holder.offsetWidth){f=u[w]-o-20;p=SVG.parent_holder.offsetHeight-q[w]+5;A.setAttribute("d","M "+(f-5)+" "+(p+5)+" L "+(f-5)+" "+(p-d+5)+" L "+(f+o+5)+" "+(p-d+5)+" L "+(f+o+5)+" "+(p-d/2+2.5)+" L "+(f+o+10)+" "+(p-d/2+5)+" L "+(f+o+5)+" "+(p-d/2+7.5)+" L "+(f+o+5)+" "+(p+5)+" Z")}else{if(u[w]-k.getBoundingClientRect().width/2<0){f=u[w]+20;p=SVG.parent_holder.offsetHeight-q[w]+5;A.setAttribute("d","M "+(f-5)+" "+(p+5)+" L "+(f-5)+" "+(p-d/2+7.5)+" L "+(f-10)+" "+(p-d/2+5)+" L "+(f-5)+" "+(p-d/2+2.5)+" L "+(f-5)+" "+(p-d+5)+" L "+(f+o+5)+" "+(p-d+5)+" L "+(f+o+5)+" "+(p+5)+" Z")}else{A.setAttribute("d","M "+(f-5)+" "+(p+5)+" L "+(f-5)+" "+(p-d+5)+" L "+(f+o+5)+" "+(p-d+5)+" L "+(f+o+5)+" "+(p+5)+" L "+(f+o/2+2.5)+" "+(p+5)+" L "+(f+o/2)+" "+(p+10)+" L "+(f+o/2-2.5)+" "+(p+5)+" Z")}}k.setAttribute("x",f);k.setAttribute("y",p);l.setAttribute("display","none")}}for(w=0;w<SVG.raw_points.length;w++){s=document.createElementNS(SVG.ns,"rect");s.setAttribute("class","over");s.setAttribute("id","over_"+w);if(w==0){s.setAttribute("x",0)}else{s.setAttribute("x",(u[w]+u[w-1])/2)}s.setAttribute("y",0);s.setAttribute("fill","white");s.setAttribute("opacity","0");if(w==SVG.raw_points.length-1){s.setAttribute("width",(u[w]-u[w-1])/2+SVG.marginLeft)}else{if(w==0){s.setAttribute("width",(u[1]-u[0])/2+SVG.marginLeft)}else{s.setAttribute("width",(u[w+1]-u[w-1])/2)}}s.setAttribute("height","100%");SVG.g.appendChild(s);s.addEventListener("mouseover",function(){SVG.holder.getElementById(this.getAttribute("id").replace("over","point")).setAttribute("r","6");if(SVG.labels[parseInt(this.getAttribute("id").replace("over_",""))]!==""){SVG.holder.getElementById(this.getAttribute("id").replace("over","label")).setAttribute("display","block")}});s.addEventListener("mouseout",function(){SVG.holder.getElementById(this.getAttribute("id").replace("over","point")).setAttribute("r","4");if(SVG.labels[parseInt(this.getAttribute("id").replace("over_",""))]!==""){SVG.holder.getElementById(this.getAttribute("id").replace("over","label")).setAttribute("display","none")}})}};var old=window.onresize||function(){};window.onresize=function(){old();if(SVG.g!==false){SVG.g.setAttribute("transform","translate(0, "+SVG.parent_holder.offsetHeight+") scale(1, -1)");if(SVG.x_axis===true){SVG.axis.setAttribute("x2",SVG.parent_holder.offsetWidth-13-SVG.marginRight)}[].forEach.call(SVG.holder.querySelectorAll(".label, .over, .point, .line, .graph"),function(d){d.parentNode.removeChild(d)});SVG.draw()}};