From b9b0ce110c58371d51a2d90f383eacb46dd8a9f0 Mon Sep 17 00:00:00 2001 From: Letum Date: Wed, 22 Jan 2020 09:07:09 +0100 Subject: [PATCH 01/36] add Charts.js for some beautiful charts --- src/components/Chart.bundle.min.js | 7 +++++++ src/manifest.json | 1 + 2 files changed, 8 insertions(+) create mode 100644 src/components/Chart.bundle.min.js diff --git a/src/components/Chart.bundle.min.js b/src/components/Chart.bundle.min.js new file mode 100644 index 0000000..55d9eb0 --- /dev/null +++ b/src/components/Chart.bundle.min.js @@ -0,0 +1,7 @@ +/*! + * Chart.js v2.9.3 + * https://www.chartjs.org + * (c) 2019 Chart.js Contributors + * Released under the MIT License + */ +!function(t,e){"object"==typeof exports&&"undefined"!=typeof module?module.exports=e():"function"==typeof define&&define.amd?define(e):(t=t||self).Chart=e()}(this,(function(){"use strict";"undefined"!=typeof globalThis?globalThis:"undefined"!=typeof window?window:"undefined"!=typeof global?global:"undefined"!=typeof self&&self;function t(){throw new Error("Dynamic requires are not currently supported by rollup-plugin-commonjs")}function e(t,e){return t(e={exports:{}},e.exports),e.exports}var n={aliceblue:[240,248,255],antiquewhite:[250,235,215],aqua:[0,255,255],aquamarine:[127,255,212],azure:[240,255,255],beige:[245,245,220],bisque:[255,228,196],black:[0,0,0],blanchedalmond:[255,235,205],blue:[0,0,255],blueviolet:[138,43,226],brown:[165,42,42],burlywood:[222,184,135],cadetblue:[95,158,160],chartreuse:[127,255,0],chocolate:[210,105,30],coral:[255,127,80],cornflowerblue:[100,149,237],cornsilk:[255,248,220],crimson:[220,20,60],cyan:[0,255,255],darkblue:[0,0,139],darkcyan:[0,139,139],darkgoldenrod:[184,134,11],darkgray:[169,169,169],darkgreen:[0,100,0],darkgrey:[169,169,169],darkkhaki:[189,183,107],darkmagenta:[139,0,139],darkolivegreen:[85,107,47],darkorange:[255,140,0],darkorchid:[153,50,204],darkred:[139,0,0],darksalmon:[233,150,122],darkseagreen:[143,188,143],darkslateblue:[72,61,139],darkslategray:[47,79,79],darkslategrey:[47,79,79],darkturquoise:[0,206,209],darkviolet:[148,0,211],deeppink:[255,20,147],deepskyblue:[0,191,255],dimgray:[105,105,105],dimgrey:[105,105,105],dodgerblue:[30,144,255],firebrick:[178,34,34],floralwhite:[255,250,240],forestgreen:[34,139,34],fuchsia:[255,0,255],gainsboro:[220,220,220],ghostwhite:[248,248,255],gold:[255,215,0],goldenrod:[218,165,32],gray:[128,128,128],green:[0,128,0],greenyellow:[173,255,47],grey:[128,128,128],honeydew:[240,255,240],hotpink:[255,105,180],indianred:[205,92,92],indigo:[75,0,130],ivory:[255,255,240],khaki:[240,230,140],lavender:[230,230,250],lavenderblush:[255,240,245],lawngreen:[124,252,0],lemonchiffon:[255,250,205],lightblue:[173,216,230],lightcoral:[240,128,128],lightcyan:[224,255,255],lightgoldenrodyellow:[250,250,210],lightgray:[211,211,211],lightgreen:[144,238,144],lightgrey:[211,211,211],lightpink:[255,182,193],lightsalmon:[255,160,122],lightseagreen:[32,178,170],lightskyblue:[135,206,250],lightslategray:[119,136,153],lightslategrey:[119,136,153],lightsteelblue:[176,196,222],lightyellow:[255,255,224],lime:[0,255,0],limegreen:[50,205,50],linen:[250,240,230],magenta:[255,0,255],maroon:[128,0,0],mediumaquamarine:[102,205,170],mediumblue:[0,0,205],mediumorchid:[186,85,211],mediumpurple:[147,112,219],mediumseagreen:[60,179,113],mediumslateblue:[123,104,238],mediumspringgreen:[0,250,154],mediumturquoise:[72,209,204],mediumvioletred:[199,21,133],midnightblue:[25,25,112],mintcream:[245,255,250],mistyrose:[255,228,225],moccasin:[255,228,181],navajowhite:[255,222,173],navy:[0,0,128],oldlace:[253,245,230],olive:[128,128,0],olivedrab:[107,142,35],orange:[255,165,0],orangered:[255,69,0],orchid:[218,112,214],palegoldenrod:[238,232,170],palegreen:[152,251,152],paleturquoise:[175,238,238],palevioletred:[219,112,147],papayawhip:[255,239,213],peachpuff:[255,218,185],peru:[205,133,63],pink:[255,192,203],plum:[221,160,221],powderblue:[176,224,230],purple:[128,0,128],rebeccapurple:[102,51,153],red:[255,0,0],rosybrown:[188,143,143],royalblue:[65,105,225],saddlebrown:[139,69,19],salmon:[250,128,114],sandybrown:[244,164,96],seagreen:[46,139,87],seashell:[255,245,238],sienna:[160,82,45],silver:[192,192,192],skyblue:[135,206,235],slateblue:[106,90,205],slategray:[112,128,144],slategrey:[112,128,144],snow:[255,250,250],springgreen:[0,255,127],steelblue:[70,130,180],tan:[210,180,140],teal:[0,128,128],thistle:[216,191,216],tomato:[255,99,71],turquoise:[64,224,208],violet:[238,130,238],wheat:[245,222,179],white:[255,255,255],whitesmoke:[245,245,245],yellow:[255,255,0],yellowgreen:[154,205,50]},i=e((function(t){var e={};for(var i in n)n.hasOwnProperty(i)&&(e[n[i]]=i);var a=t.exports={rgb:{channels:3,labels:"rgb"},hsl:{channels:3,labels:"hsl"},hsv:{channels:3,labels:"hsv"},hwb:{channels:3,labels:"hwb"},cmyk:{channels:4,labels:"cmyk"},xyz:{channels:3,labels:"xyz"},lab:{channels:3,labels:"lab"},lch:{channels:3,labels:"lch"},hex:{channels:1,labels:["hex"]},keyword:{channels:1,labels:["keyword"]},ansi16:{channels:1,labels:["ansi16"]},ansi256:{channels:1,labels:["ansi256"]},hcg:{channels:3,labels:["h","c","g"]},apple:{channels:3,labels:["r16","g16","b16"]},gray:{channels:1,labels:["gray"]}};for(var r in a)if(a.hasOwnProperty(r)){if(!("channels"in a[r]))throw new Error("missing channels property: "+r);if(!("labels"in a[r]))throw new Error("missing channel labels property: "+r);if(a[r].labels.length!==a[r].channels)throw new Error("channel and label counts mismatch: "+r);var o=a[r].channels,s=a[r].labels;delete a[r].channels,delete a[r].labels,Object.defineProperty(a[r],"channels",{value:o}),Object.defineProperty(a[r],"labels",{value:s})}a.rgb.hsl=function(t){var e,n,i=t[0]/255,a=t[1]/255,r=t[2]/255,o=Math.min(i,a,r),s=Math.max(i,a,r),l=s-o;return s===o?e=0:i===s?e=(a-r)/l:a===s?e=2+(r-i)/l:r===s&&(e=4+(i-a)/l),(e=Math.min(60*e,360))<0&&(e+=360),n=(o+s)/2,[e,100*(s===o?0:n<=.5?l/(s+o):l/(2-s-o)),100*n]},a.rgb.hsv=function(t){var e,n,i,a,r,o=t[0]/255,s=t[1]/255,l=t[2]/255,u=Math.max(o,s,l),d=u-Math.min(o,s,l),h=function(t){return(u-t)/6/d+.5};return 0===d?a=r=0:(r=d/u,e=h(o),n=h(s),i=h(l),o===u?a=i-n:s===u?a=1/3+e-i:l===u&&(a=2/3+n-e),a<0?a+=1:a>1&&(a-=1)),[360*a,100*r,100*u]},a.rgb.hwb=function(t){var e=t[0],n=t[1],i=t[2];return[a.rgb.hsl(t)[0],100*(1/255*Math.min(e,Math.min(n,i))),100*(i=1-1/255*Math.max(e,Math.max(n,i)))]},a.rgb.cmyk=function(t){var e,n=t[0]/255,i=t[1]/255,a=t[2]/255;return[100*((1-n-(e=Math.min(1-n,1-i,1-a)))/(1-e)||0),100*((1-i-e)/(1-e)||0),100*((1-a-e)/(1-e)||0),100*e]},a.rgb.keyword=function(t){var i=e[t];if(i)return i;var a,r,o,s=1/0;for(var l in n)if(n.hasOwnProperty(l)){var u=n[l],d=(r=t,o=u,Math.pow(r[0]-o[0],2)+Math.pow(r[1]-o[1],2)+Math.pow(r[2]-o[2],2));d.04045?Math.pow((e+.055)/1.055,2.4):e/12.92)+.3576*(n=n>.04045?Math.pow((n+.055)/1.055,2.4):n/12.92)+.1805*(i=i>.04045?Math.pow((i+.055)/1.055,2.4):i/12.92)),100*(.2126*e+.7152*n+.0722*i),100*(.0193*e+.1192*n+.9505*i)]},a.rgb.lab=function(t){var e=a.rgb.xyz(t),n=e[0],i=e[1],r=e[2];return i/=100,r/=108.883,n=(n/=95.047)>.008856?Math.pow(n,1/3):7.787*n+16/116,[116*(i=i>.008856?Math.pow(i,1/3):7.787*i+16/116)-16,500*(n-i),200*(i-(r=r>.008856?Math.pow(r,1/3):7.787*r+16/116))]},a.hsl.rgb=function(t){var e,n,i,a,r,o=t[0]/360,s=t[1]/100,l=t[2]/100;if(0===s)return[r=255*l,r,r];e=2*l-(n=l<.5?l*(1+s):l+s-l*s),a=[0,0,0];for(var u=0;u<3;u++)(i=o+1/3*-(u-1))<0&&i++,i>1&&i--,r=6*i<1?e+6*(n-e)*i:2*i<1?n:3*i<2?e+(n-e)*(2/3-i)*6:e,a[u]=255*r;return a},a.hsl.hsv=function(t){var e=t[0],n=t[1]/100,i=t[2]/100,a=n,r=Math.max(i,.01);return n*=(i*=2)<=1?i:2-i,a*=r<=1?r:2-r,[e,100*(0===i?2*a/(r+a):2*n/(i+n)),100*((i+n)/2)]},a.hsv.rgb=function(t){var e=t[0]/60,n=t[1]/100,i=t[2]/100,a=Math.floor(e)%6,r=e-Math.floor(e),o=255*i*(1-n),s=255*i*(1-n*r),l=255*i*(1-n*(1-r));switch(i*=255,a){case 0:return[i,l,o];case 1:return[s,i,o];case 2:return[o,i,l];case 3:return[o,s,i];case 4:return[l,o,i];case 5:return[i,o,s]}},a.hsv.hsl=function(t){var e,n,i,a=t[0],r=t[1]/100,o=t[2]/100,s=Math.max(o,.01);return i=(2-r)*o,n=r*s,[a,100*(n=(n/=(e=(2-r)*s)<=1?e:2-e)||0),100*(i/=2)]},a.hwb.rgb=function(t){var e,n,i,a,r,o,s,l=t[0]/360,u=t[1]/100,d=t[2]/100,h=u+d;switch(h>1&&(u/=h,d/=h),i=6*l-(e=Math.floor(6*l)),0!=(1&e)&&(i=1-i),a=u+i*((n=1-d)-u),e){default:case 6:case 0:r=n,o=a,s=u;break;case 1:r=a,o=n,s=u;break;case 2:r=u,o=n,s=a;break;case 3:r=u,o=a,s=n;break;case 4:r=a,o=u,s=n;break;case 5:r=n,o=u,s=a}return[255*r,255*o,255*s]},a.cmyk.rgb=function(t){var e=t[0]/100,n=t[1]/100,i=t[2]/100,a=t[3]/100;return[255*(1-Math.min(1,e*(1-a)+a)),255*(1-Math.min(1,n*(1-a)+a)),255*(1-Math.min(1,i*(1-a)+a))]},a.xyz.rgb=function(t){var e,n,i,a=t[0]/100,r=t[1]/100,o=t[2]/100;return n=-.9689*a+1.8758*r+.0415*o,i=.0557*a+-.204*r+1.057*o,e=(e=3.2406*a+-1.5372*r+-.4986*o)>.0031308?1.055*Math.pow(e,1/2.4)-.055:12.92*e,n=n>.0031308?1.055*Math.pow(n,1/2.4)-.055:12.92*n,i=i>.0031308?1.055*Math.pow(i,1/2.4)-.055:12.92*i,[255*(e=Math.min(Math.max(0,e),1)),255*(n=Math.min(Math.max(0,n),1)),255*(i=Math.min(Math.max(0,i),1))]},a.xyz.lab=function(t){var e=t[0],n=t[1],i=t[2];return n/=100,i/=108.883,e=(e/=95.047)>.008856?Math.pow(e,1/3):7.787*e+16/116,[116*(n=n>.008856?Math.pow(n,1/3):7.787*n+16/116)-16,500*(e-n),200*(n-(i=i>.008856?Math.pow(i,1/3):7.787*i+16/116))]},a.lab.xyz=function(t){var e,n,i,a=t[0];e=t[1]/500+(n=(a+16)/116),i=n-t[2]/200;var r=Math.pow(n,3),o=Math.pow(e,3),s=Math.pow(i,3);return n=r>.008856?r:(n-16/116)/7.787,e=o>.008856?o:(e-16/116)/7.787,i=s>.008856?s:(i-16/116)/7.787,[e*=95.047,n*=100,i*=108.883]},a.lab.lch=function(t){var e,n=t[0],i=t[1],a=t[2];return(e=360*Math.atan2(a,i)/2/Math.PI)<0&&(e+=360),[n,Math.sqrt(i*i+a*a),e]},a.lch.lab=function(t){var e,n=t[0],i=t[1];return e=t[2]/360*2*Math.PI,[n,i*Math.cos(e),i*Math.sin(e)]},a.rgb.ansi16=function(t){var e=t[0],n=t[1],i=t[2],r=1 in arguments?arguments[1]:a.rgb.hsv(t)[2];if(0===(r=Math.round(r/50)))return 30;var o=30+(Math.round(i/255)<<2|Math.round(n/255)<<1|Math.round(e/255));return 2===r&&(o+=60),o},a.hsv.ansi16=function(t){return a.rgb.ansi16(a.hsv.rgb(t),t[2])},a.rgb.ansi256=function(t){var e=t[0],n=t[1],i=t[2];return e===n&&n===i?e<8?16:e>248?231:Math.round((e-8)/247*24)+232:16+36*Math.round(e/255*5)+6*Math.round(n/255*5)+Math.round(i/255*5)},a.ansi16.rgb=function(t){var e=t%10;if(0===e||7===e)return t>50&&(e+=3.5),[e=e/10.5*255,e,e];var n=.5*(1+~~(t>50));return[(1&e)*n*255,(e>>1&1)*n*255,(e>>2&1)*n*255]},a.ansi256.rgb=function(t){if(t>=232){var e=10*(t-232)+8;return[e,e,e]}var n;return t-=16,[Math.floor(t/36)/5*255,Math.floor((n=t%36)/6)/5*255,n%6/5*255]},a.rgb.hex=function(t){var e=(((255&Math.round(t[0]))<<16)+((255&Math.round(t[1]))<<8)+(255&Math.round(t[2]))).toString(16).toUpperCase();return"000000".substring(e.length)+e},a.hex.rgb=function(t){var e=t.toString(16).match(/[a-f0-9]{6}|[a-f0-9]{3}/i);if(!e)return[0,0,0];var n=e[0];3===e[0].length&&(n=n.split("").map((function(t){return t+t})).join(""));var i=parseInt(n,16);return[i>>16&255,i>>8&255,255&i]},a.rgb.hcg=function(t){var e,n=t[0]/255,i=t[1]/255,a=t[2]/255,r=Math.max(Math.max(n,i),a),o=Math.min(Math.min(n,i),a),s=r-o;return e=s<=0?0:r===n?(i-a)/s%6:r===i?2+(a-n)/s:4+(n-i)/s+4,e/=6,[360*(e%=1),100*s,100*(s<1?o/(1-s):0)]},a.hsl.hcg=function(t){var e=t[1]/100,n=t[2]/100,i=1,a=0;return(i=n<.5?2*e*n:2*e*(1-n))<1&&(a=(n-.5*i)/(1-i)),[t[0],100*i,100*a]},a.hsv.hcg=function(t){var e=t[1]/100,n=t[2]/100,i=e*n,a=0;return i<1&&(a=(n-i)/(1-i)),[t[0],100*i,100*a]},a.hcg.rgb=function(t){var e=t[0]/360,n=t[1]/100,i=t[2]/100;if(0===n)return[255*i,255*i,255*i];var a,r=[0,0,0],o=e%1*6,s=o%1,l=1-s;switch(Math.floor(o)){case 0:r[0]=1,r[1]=s,r[2]=0;break;case 1:r[0]=l,r[1]=1,r[2]=0;break;case 2:r[0]=0,r[1]=1,r[2]=s;break;case 3:r[0]=0,r[1]=l,r[2]=1;break;case 4:r[0]=s,r[1]=0,r[2]=1;break;default:r[0]=1,r[1]=0,r[2]=l}return a=(1-n)*i,[255*(n*r[0]+a),255*(n*r[1]+a),255*(n*r[2]+a)]},a.hcg.hsv=function(t){var e=t[1]/100,n=e+t[2]/100*(1-e),i=0;return n>0&&(i=e/n),[t[0],100*i,100*n]},a.hcg.hsl=function(t){var e=t[1]/100,n=t[2]/100*(1-e)+.5*e,i=0;return n>0&&n<.5?i=e/(2*n):n>=.5&&n<1&&(i=e/(2*(1-n))),[t[0],100*i,100*n]},a.hcg.hwb=function(t){var e=t[1]/100,n=e+t[2]/100*(1-e);return[t[0],100*(n-e),100*(1-n)]},a.hwb.hcg=function(t){var e=t[1]/100,n=1-t[2]/100,i=n-e,a=0;return i<1&&(a=(n-i)/(1-i)),[t[0],100*i,100*a]},a.apple.rgb=function(t){return[t[0]/65535*255,t[1]/65535*255,t[2]/65535*255]},a.rgb.apple=function(t){return[t[0]/255*65535,t[1]/255*65535,t[2]/255*65535]},a.gray.rgb=function(t){return[t[0]/100*255,t[0]/100*255,t[0]/100*255]},a.gray.hsl=a.gray.hsv=function(t){return[0,0,t[0]]},a.gray.hwb=function(t){return[0,100,t[0]]},a.gray.cmyk=function(t){return[0,0,0,t[0]]},a.gray.lab=function(t){return[t[0],0,0]},a.gray.hex=function(t){var e=255&Math.round(t[0]/100*255),n=((e<<16)+(e<<8)+e).toString(16).toUpperCase();return"000000".substring(n.length)+n},a.rgb.gray=function(t){return[(t[0]+t[1]+t[2])/3/255*100]}}));i.rgb,i.hsl,i.hsv,i.hwb,i.cmyk,i.xyz,i.lab,i.lch,i.hex,i.keyword,i.ansi16,i.ansi256,i.hcg,i.apple,i.gray;function a(t){var e=function(){for(var t={},e=Object.keys(i),n=e.length,a=0;a1&&(e=Array.prototype.slice.call(arguments));var n=t(e);if("object"==typeof n)for(var i=n.length,a=0;a1&&(e=Array.prototype.slice.call(arguments)),t(e))};return"conversion"in t&&(e.conversion=t.conversion),e}(i)}))}));var l=s,u={aliceblue:[240,248,255],antiquewhite:[250,235,215],aqua:[0,255,255],aquamarine:[127,255,212],azure:[240,255,255],beige:[245,245,220],bisque:[255,228,196],black:[0,0,0],blanchedalmond:[255,235,205],blue:[0,0,255],blueviolet:[138,43,226],brown:[165,42,42],burlywood:[222,184,135],cadetblue:[95,158,160],chartreuse:[127,255,0],chocolate:[210,105,30],coral:[255,127,80],cornflowerblue:[100,149,237],cornsilk:[255,248,220],crimson:[220,20,60],cyan:[0,255,255],darkblue:[0,0,139],darkcyan:[0,139,139],darkgoldenrod:[184,134,11],darkgray:[169,169,169],darkgreen:[0,100,0],darkgrey:[169,169,169],darkkhaki:[189,183,107],darkmagenta:[139,0,139],darkolivegreen:[85,107,47],darkorange:[255,140,0],darkorchid:[153,50,204],darkred:[139,0,0],darksalmon:[233,150,122],darkseagreen:[143,188,143],darkslateblue:[72,61,139],darkslategray:[47,79,79],darkslategrey:[47,79,79],darkturquoise:[0,206,209],darkviolet:[148,0,211],deeppink:[255,20,147],deepskyblue:[0,191,255],dimgray:[105,105,105],dimgrey:[105,105,105],dodgerblue:[30,144,255],firebrick:[178,34,34],floralwhite:[255,250,240],forestgreen:[34,139,34],fuchsia:[255,0,255],gainsboro:[220,220,220],ghostwhite:[248,248,255],gold:[255,215,0],goldenrod:[218,165,32],gray:[128,128,128],green:[0,128,0],greenyellow:[173,255,47],grey:[128,128,128],honeydew:[240,255,240],hotpink:[255,105,180],indianred:[205,92,92],indigo:[75,0,130],ivory:[255,255,240],khaki:[240,230,140],lavender:[230,230,250],lavenderblush:[255,240,245],lawngreen:[124,252,0],lemonchiffon:[255,250,205],lightblue:[173,216,230],lightcoral:[240,128,128],lightcyan:[224,255,255],lightgoldenrodyellow:[250,250,210],lightgray:[211,211,211],lightgreen:[144,238,144],lightgrey:[211,211,211],lightpink:[255,182,193],lightsalmon:[255,160,122],lightseagreen:[32,178,170],lightskyblue:[135,206,250],lightslategray:[119,136,153],lightslategrey:[119,136,153],lightsteelblue:[176,196,222],lightyellow:[255,255,224],lime:[0,255,0],limegreen:[50,205,50],linen:[250,240,230],magenta:[255,0,255],maroon:[128,0,0],mediumaquamarine:[102,205,170],mediumblue:[0,0,205],mediumorchid:[186,85,211],mediumpurple:[147,112,219],mediumseagreen:[60,179,113],mediumslateblue:[123,104,238],mediumspringgreen:[0,250,154],mediumturquoise:[72,209,204],mediumvioletred:[199,21,133],midnightblue:[25,25,112],mintcream:[245,255,250],mistyrose:[255,228,225],moccasin:[255,228,181],navajowhite:[255,222,173],navy:[0,0,128],oldlace:[253,245,230],olive:[128,128,0],olivedrab:[107,142,35],orange:[255,165,0],orangered:[255,69,0],orchid:[218,112,214],palegoldenrod:[238,232,170],palegreen:[152,251,152],paleturquoise:[175,238,238],palevioletred:[219,112,147],papayawhip:[255,239,213],peachpuff:[255,218,185],peru:[205,133,63],pink:[255,192,203],plum:[221,160,221],powderblue:[176,224,230],purple:[128,0,128],rebeccapurple:[102,51,153],red:[255,0,0],rosybrown:[188,143,143],royalblue:[65,105,225],saddlebrown:[139,69,19],salmon:[250,128,114],sandybrown:[244,164,96],seagreen:[46,139,87],seashell:[255,245,238],sienna:[160,82,45],silver:[192,192,192],skyblue:[135,206,235],slateblue:[106,90,205],slategray:[112,128,144],slategrey:[112,128,144],snow:[255,250,250],springgreen:[0,255,127],steelblue:[70,130,180],tan:[210,180,140],teal:[0,128,128],thistle:[216,191,216],tomato:[255,99,71],turquoise:[64,224,208],violet:[238,130,238],wheat:[245,222,179],white:[255,255,255],whitesmoke:[245,245,245],yellow:[255,255,0],yellowgreen:[154,205,50]},d={getRgba:h,getHsla:c,getRgb:function(t){var e=h(t);return e&&e.slice(0,3)},getHsl:function(t){var e=c(t);return e&&e.slice(0,3)},getHwb:f,getAlpha:function(t){var e=h(t);if(e)return e[3];if(e=c(t))return e[3];if(e=f(t))return e[3]},hexString:function(t,e){e=void 0!==e&&3===t.length?e:t[3];return"#"+b(t[0])+b(t[1])+b(t[2])+(e>=0&&e<1?b(Math.round(255*e)):"")},rgbString:function(t,e){if(e<1||t[3]&&t[3]<1)return g(t,e);return"rgb("+t[0]+", "+t[1]+", "+t[2]+")"},rgbaString:g,percentString:function(t,e){if(e<1||t[3]&&t[3]<1)return m(t,e);var n=Math.round(t[0]/255*100),i=Math.round(t[1]/255*100),a=Math.round(t[2]/255*100);return"rgb("+n+"%, "+i+"%, "+a+"%)"},percentaString:m,hslString:function(t,e){if(e<1||t[3]&&t[3]<1)return p(t,e);return"hsl("+t[0]+", "+t[1]+"%, "+t[2]+"%)"},hslaString:p,hwbString:function(t,e){void 0===e&&(e=void 0!==t[3]?t[3]:1);return"hwb("+t[0]+", "+t[1]+"%, "+t[2]+"%"+(void 0!==e&&1!==e?", "+e:"")+")"},keyword:function(t){return y[t.slice(0,3)]}};function h(t){if(t){var e=[0,0,0],n=1,i=t.match(/^#([a-fA-F0-9]{3,4})$/i),a="";if(i){a=(i=i[1])[3];for(var r=0;rn?(e+.05)/(n+.05):(n+.05)/(e+.05)},level:function(t){var e=this.contrast(t);return e>=7.1?"AAA":e>=4.5?"AA":""},dark:function(){var t=this.values.rgb;return(299*t[0]+587*t[1]+114*t[2])/1e3<128},light:function(){return!this.dark()},negate:function(){for(var t=[],e=0;e<3;e++)t[e]=255-this.values.rgb[e];return this.setValues("rgb",t),this},lighten:function(t){var e=this.values.hsl;return e[2]+=e[2]*t,this.setValues("hsl",e),this},darken:function(t){var e=this.values.hsl;return e[2]-=e[2]*t,this.setValues("hsl",e),this},saturate:function(t){var e=this.values.hsl;return e[1]+=e[1]*t,this.setValues("hsl",e),this},desaturate:function(t){var e=this.values.hsl;return e[1]-=e[1]*t,this.setValues("hsl",e),this},whiten:function(t){var e=this.values.hwb;return e[1]+=e[1]*t,this.setValues("hwb",e),this},blacken:function(t){var e=this.values.hwb;return e[2]+=e[2]*t,this.setValues("hwb",e),this},greyscale:function(){var t=this.values.rgb,e=.3*t[0]+.59*t[1]+.11*t[2];return this.setValues("rgb",[e,e,e]),this},clearer:function(t){var e=this.values.alpha;return this.setValues("alpha",e-e*t),this},opaquer:function(t){var e=this.values.alpha;return this.setValues("alpha",e+e*t),this},rotate:function(t){var e=this.values.hsl,n=(e[0]+t)%360;return e[0]=n<0?360+n:n,this.setValues("hsl",e),this},mix:function(t,e){var n=t,i=void 0===e?.5:e,a=2*i-1,r=this.alpha()-n.alpha(),o=((a*r==-1?a:(a+r)/(1+a*r))+1)/2,s=1-o;return this.rgb(o*this.red()+s*n.red(),o*this.green()+s*n.green(),o*this.blue()+s*n.blue()).alpha(this.alpha()*i+n.alpha()*(1-i))},toJSON:function(){return this.rgb()},clone:function(){var t,e,n=new _,i=this.values,a=n.values;for(var r in i)i.hasOwnProperty(r)&&(t=i[r],"[object Array]"===(e={}.toString.call(t))?a[r]=t.slice(0):"[object Number]"===e?a[r]=t:console.error("unexpected color value:",t));return n}},_.prototype.spaces={rgb:["red","green","blue"],hsl:["hue","saturation","lightness"],hsv:["hue","saturation","value"],hwb:["hue","whiteness","blackness"],cmyk:["cyan","magenta","yellow","black"]},_.prototype.maxes={rgb:[255,255,255],hsl:[360,100,100],hsv:[360,100,100],hwb:[360,100,100],cmyk:[100,100,100,100]},_.prototype.getValues=function(t){for(var e=this.values,n={},i=0;i=0;a--)e.call(n,t[a],a);else for(a=0;a=1?t:-(Math.sqrt(1-t*t)-1)},easeOutCirc:function(t){return Math.sqrt(1-(t-=1)*t)},easeInOutCirc:function(t){return(t/=.5)<1?-.5*(Math.sqrt(1-t*t)-1):.5*(Math.sqrt(1-(t-=2)*t)+1)},easeInElastic:function(t){var e=1.70158,n=0,i=1;return 0===t?0:1===t?1:(n||(n=.3),i<1?(i=1,e=n/4):e=n/(2*Math.PI)*Math.asin(1/i),-i*Math.pow(2,10*(t-=1))*Math.sin((t-e)*(2*Math.PI)/n))},easeOutElastic:function(t){var e=1.70158,n=0,i=1;return 0===t?0:1===t?1:(n||(n=.3),i<1?(i=1,e=n/4):e=n/(2*Math.PI)*Math.asin(1/i),i*Math.pow(2,-10*t)*Math.sin((t-e)*(2*Math.PI)/n)+1)},easeInOutElastic:function(t){var e=1.70158,n=0,i=1;return 0===t?0:2==(t/=.5)?1:(n||(n=.45),i<1?(i=1,e=n/4):e=n/(2*Math.PI)*Math.asin(1/i),t<1?i*Math.pow(2,10*(t-=1))*Math.sin((t-e)*(2*Math.PI)/n)*-.5:i*Math.pow(2,-10*(t-=1))*Math.sin((t-e)*(2*Math.PI)/n)*.5+1)},easeInBack:function(t){var e=1.70158;return t*t*((e+1)*t-e)},easeOutBack:function(t){var e=1.70158;return(t-=1)*t*((e+1)*t+e)+1},easeInOutBack:function(t){var e=1.70158;return(t/=.5)<1?t*t*((1+(e*=1.525))*t-e)*.5:.5*((t-=2)*t*((1+(e*=1.525))*t+e)+2)},easeInBounce:function(t){return 1-D.easeOutBounce(1-t)},easeOutBounce:function(t){return t<1/2.75?7.5625*t*t:t<2/2.75?7.5625*(t-=1.5/2.75)*t+.75:t<2.5/2.75?7.5625*(t-=2.25/2.75)*t+.9375:7.5625*(t-=2.625/2.75)*t+.984375},easeInOutBounce:function(t){return t<.5?.5*D.easeInBounce(2*t):.5*D.easeOutBounce(2*t-1)+.5}},C={effects:D};S.easingEffects=D;var P=Math.PI,T=P/180,O=2*P,A=P/2,F=P/4,I=2*P/3,L={clear:function(t){t.ctx.clearRect(0,0,t.width,t.height)},roundedRect:function(t,e,n,i,a,r){if(r){var o=Math.min(r,a/2,i/2),s=e+o,l=n+o,u=e+i-o,d=n+a-o;t.moveTo(e,l),se.left-1e-6&&t.xe.top-1e-6&&t.y0&&this.requestAnimationFrame()},advance:function(){for(var t,e,n,i,a=this.animations,r=0;r=n?(H.callback(t.onAnimationComplete,[t],e),e.animating=!1,a.splice(r,1)):++r}},Q=H.options.resolve,tt=["push","pop","shift","splice","unshift"];function et(t,e){var n=t._chartjs;if(n){var i=n.listeners,a=i.indexOf(e);-1!==a&&i.splice(a,1),i.length>0||(tt.forEach((function(e){delete t[e]})),delete t._chartjs)}}var nt=function(t,e){this.initialize(t,e)};H.extend(nt.prototype,{datasetElementType:null,dataElementType:null,_datasetElementOptions:["backgroundColor","borderCapStyle","borderColor","borderDash","borderDashOffset","borderJoinStyle","borderWidth"],_dataElementOptions:["backgroundColor","borderColor","borderWidth","pointStyle"],initialize:function(t,e){var n=this;n.chart=t,n.index=e,n.linkScales(),n.addElements(),n._type=n.getMeta().type},updateIndex:function(t){this.index=t},linkScales:function(){var t=this.getMeta(),e=this.chart,n=e.scales,i=this.getDataset(),a=e.options.scales;null!==t.xAxisID&&t.xAxisID in n&&!i.xAxisID||(t.xAxisID=i.xAxisID||a.xAxes[0].id),null!==t.yAxisID&&t.yAxisID in n&&!i.yAxisID||(t.yAxisID=i.yAxisID||a.yAxes[0].id)},getDataset:function(){return this.chart.data.datasets[this.index]},getMeta:function(){return this.chart.getDatasetMeta(this.index)},getScaleForId:function(t){return this.chart.scales[t]},_getValueScaleId:function(){return this.getMeta().yAxisID},_getIndexScaleId:function(){return this.getMeta().xAxisID},_getValueScale:function(){return this.getScaleForId(this._getValueScaleId())},_getIndexScale:function(){return this.getScaleForId(this._getIndexScaleId())},reset:function(){this._update(!0)},destroy:function(){this._data&&et(this._data,this)},createMetaDataset:function(){var t=this.datasetElementType;return t&&new t({_chart:this.chart,_datasetIndex:this.index})},createMetaData:function(t){var e=this.dataElementType;return e&&new e({_chart:this.chart,_datasetIndex:this.index,_index:t})},addElements:function(){var t,e,n=this.getMeta(),i=this.getDataset().data||[],a=n.data;for(t=0,e=i.length;tn&&this.insertElements(n,i-n)},insertElements:function(t,e){for(var n=0;na?(r=a/e.innerRadius,t.arc(o,s,e.innerRadius-a,i+r,n-r,!0)):t.arc(o,s,a,i+Math.PI/2,n-Math.PI/2),t.closePath(),t.clip()}function ot(t,e,n){var i="inner"===e.borderAlign;i?(t.lineWidth=2*e.borderWidth,t.lineJoin="round"):(t.lineWidth=e.borderWidth,t.lineJoin="bevel"),n.fullCircles&&function(t,e,n,i){var a,r=n.endAngle;for(i&&(n.endAngle=n.startAngle+at,rt(t,n),n.endAngle=r,n.endAngle===n.startAngle&&n.fullCircles&&(n.endAngle+=at,n.fullCircles--)),t.beginPath(),t.arc(n.x,n.y,n.innerRadius,n.startAngle+at,n.startAngle,!0),a=0;as;)a-=at;for(;a=o&&a<=s,u=r>=n.innerRadius&&r<=n.outerRadius;return l&&u}return!1},getCenterPoint:function(){var t=this._view,e=(t.startAngle+t.endAngle)/2,n=(t.innerRadius+t.outerRadius)/2;return{x:t.x+Math.cos(e)*n,y:t.y+Math.sin(e)*n}},getArea:function(){var t=this._view;return Math.PI*((t.endAngle-t.startAngle)/(2*Math.PI))*(Math.pow(t.outerRadius,2)-Math.pow(t.innerRadius,2))},tooltipPosition:function(){var t=this._view,e=t.startAngle+(t.endAngle-t.startAngle)/2,n=(t.outerRadius-t.innerRadius)/2+t.innerRadius;return{x:t.x+Math.cos(e)*n,y:t.y+Math.sin(e)*n}},draw:function(){var t,e=this._chart.ctx,n=this._view,i="inner"===n.borderAlign?.33:0,a={x:n.x,y:n.y,innerRadius:n.innerRadius,outerRadius:Math.max(n.outerRadius-i,0),pixelMargin:i,startAngle:n.startAngle,endAngle:n.endAngle,fullCircles:Math.floor(n.circumference/at)};if(e.save(),e.fillStyle=n.backgroundColor,e.strokeStyle=n.borderColor,a.fullCircles){for(a.endAngle=a.startAngle+at,e.beginPath(),e.arc(a.x,a.y,a.outerRadius,a.startAngle,a.endAngle),e.arc(a.x,a.y,a.innerRadius,a.endAngle,a.startAngle,!0),e.closePath(),t=0;tt.x&&(e=bt(e,"left","right")):t.basen?n:i,r:l.right||a<0?0:a>e?e:a,b:l.bottom||r<0?0:r>n?n:r,l:l.left||o<0?0:o>e?e:o}}function xt(t,e,n){var i=null===e,a=null===n,r=!(!t||i&&a)&&vt(t);return r&&(i||e>=r.left&&e<=r.right)&&(a||n>=r.top&&n<=r.bottom)}W._set("global",{elements:{rectangle:{backgroundColor:mt,borderColor:mt,borderSkipped:"bottom",borderWidth:0}}});var _t=$.extend({_type:"rectangle",draw:function(){var t=this._chart.ctx,e=this._view,n=function(t){var e=vt(t),n=e.right-e.left,i=e.bottom-e.top,a=yt(t,n/2,i/2);return{outer:{x:e.left,y:e.top,w:n,h:i},inner:{x:e.left+a.l,y:e.top+a.t,w:n-a.l-a.r,h:i-a.t-a.b}}}(e),i=n.outer,a=n.inner;t.fillStyle=e.backgroundColor,t.fillRect(i.x,i.y,i.w,i.h),i.w===a.w&&i.h===a.h||(t.save(),t.beginPath(),t.rect(i.x,i.y,i.w,i.h),t.clip(),t.fillStyle=e.borderColor,t.rect(a.x,a.y,a.w,a.h),t.fill("evenodd"),t.restore())},height:function(){var t=this._view;return t.base-t.y},inRange:function(t,e){return xt(this._view,t,e)},inLabelRange:function(t,e){var n=this._view;return pt(n)?xt(n,t,null):xt(n,null,e)},inXRange:function(t){return xt(this._view,t,null)},inYRange:function(t){return xt(this._view,null,t)},getCenterPoint:function(){var t,e,n=this._view;return pt(n)?(t=n.x,e=(n.y+n.base)/2):(t=(n.x+n.base)/2,e=n.y),{x:t,y:e}},getArea:function(){var t=this._view;return pt(t)?t.width*Math.abs(t.y-t.base):t.height*Math.abs(t.x-t.base)},tooltipPosition:function(){var t=this._view;return{x:t.x,y:t.y}}}),wt={},kt=st,Mt=dt,St=gt,Dt=_t;wt.Arc=kt,wt.Line=Mt,wt.Point=St,wt.Rectangle=Dt;var Ct=H._deprecated,Pt=H.valueOrDefault;function Tt(t,e,n){var i,a,r=n.barThickness,o=e.stackCount,s=e.pixels[t],l=H.isNullOrUndef(r)?function(t,e){var n,i,a,r,o=t._length;for(a=1,r=e.length;a0?Math.min(o,Math.abs(i-n)):o,n=i;return o}(e.scale,e.pixels):-1;return H.isNullOrUndef(r)?(i=l*n.categoryPercentage,a=n.barPercentage):(i=r*o,a=1),{chunk:i/o,ratio:a,start:s-i/2}}W._set("bar",{hover:{mode:"label"},scales:{xAxes:[{type:"category",offset:!0,gridLines:{offsetGridLines:!0}}],yAxes:[{type:"linear"}]}}),W._set("global",{datasets:{bar:{categoryPercentage:.8,barPercentage:.9}}});var Ot=it.extend({dataElementType:wt.Rectangle,_dataElementOptions:["backgroundColor","borderColor","borderSkipped","borderWidth","barPercentage","barThickness","categoryPercentage","maxBarThickness","minBarLength"],initialize:function(){var t,e,n=this;it.prototype.initialize.apply(n,arguments),(t=n.getMeta()).stack=n.getDataset().stack,t.bar=!0,e=n._getIndexScale().options,Ct("bar chart",e.barPercentage,"scales.[x/y]Axes.barPercentage","dataset.barPercentage"),Ct("bar chart",e.barThickness,"scales.[x/y]Axes.barThickness","dataset.barThickness"),Ct("bar chart",e.categoryPercentage,"scales.[x/y]Axes.categoryPercentage","dataset.categoryPercentage"),Ct("bar chart",n._getValueScale().options.minBarLength,"scales.[x/y]Axes.minBarLength","dataset.minBarLength"),Ct("bar chart",e.maxBarThickness,"scales.[x/y]Axes.maxBarThickness","dataset.maxBarThickness")},update:function(t){var e,n,i=this.getMeta().data;for(this._ruler=this.getRuler(),e=0,n=i.length;e=0&&m.min>=0?m.min:m.max,x=void 0===m.start?m.end:m.max>=0&&m.min>=0?m.max-m.min:m.min-m.max,_=g.length;if(v||void 0===v&&void 0!==b)for(i=0;i<_&&(a=g[i]).index!==t;++i)a.stack===b&&(r=void 0===(u=h._parseValue(f[a.index].data[e])).start?u.end:u.min>=0&&u.max>=0?u.max:u.min,(m.min<0&&r<0||m.max>=0&&r>0)&&(y+=r));return o=h.getPixelForValue(y),l=(s=h.getPixelForValue(y+x))-o,void 0!==p&&Math.abs(l)=0&&!c||x<0&&c?o-p:o+p),{size:l,base:o,head:s,center:s+l/2}},calculateBarIndexPixels:function(t,e,n,i){var a="flex"===i.barThickness?function(t,e,n){var i,a=e.pixels,r=a[t],o=t>0?a[t-1]:null,s=t=Rt?-Nt:b<-Rt?Nt:0)+p,x=Math.cos(b),_=Math.sin(b),w=Math.cos(y),k=Math.sin(y),M=b<=0&&y>=0||y>=Nt,S=b<=Wt&&y>=Wt||y>=Nt+Wt,D=b<=-Wt&&y>=-Wt||y>=Rt+Wt,C=b===-Rt||y>=Rt?-1:Math.min(x,x*m,w,w*m),P=D?-1:Math.min(_,_*m,k,k*m),T=M?1:Math.max(x,x*m,w,w*m),O=S?1:Math.max(_,_*m,k,k*m);u=(T-C)/2,d=(O-P)/2,h=-(T+C)/2,c=-(O+P)/2}for(i=0,a=g.length;i0&&!isNaN(t)?Nt*(Math.abs(t)/e):0},getMaxBorderWidth:function(t){var e,n,i,a,r,o,s,l,u=0,d=this.chart;if(!t)for(e=0,n=d.data.datasets.length;e(u=s>u?s:u)?l:u);return u},setHoverStyle:function(t){var e=t._model,n=t._options,i=H.getHoverColor;t.$previousStyle={backgroundColor:e.backgroundColor,borderColor:e.borderColor,borderWidth:e.borderWidth},e.backgroundColor=Lt(n.hoverBackgroundColor,i(n.backgroundColor)),e.borderColor=Lt(n.hoverBorderColor,i(n.borderColor)),e.borderWidth=Lt(n.hoverBorderWidth,n.borderWidth)},_getRingWeightOffset:function(t){for(var e=0,n=0;n0&&Ht(l[t-1]._model,s)&&(n.controlPointPreviousX=u(n.controlPointPreviousX,s.left,s.right),n.controlPointPreviousY=u(n.controlPointPreviousY,s.top,s.bottom)),t0&&(r=t.getDatasetMeta(r[0]._datasetIndex).data),r},"x-axis":function(t,e){return ae(t,e,{intersect:!1})},point:function(t,e){return ee(t,Qt(e,t))},nearest:function(t,e,n){var i=Qt(e,t);n.axis=n.axis||"xy";var a=ie(n.axis);return ne(t,i,n.intersect,a)},x:function(t,e,n){var i=Qt(e,t),a=[],r=!1;return te(t,(function(t){t.inXRange(i.x)&&a.push(t),t.inRange(i.x,i.y)&&(r=!0)})),n.intersect&&!r&&(a=[]),a},y:function(t,e,n){var i=Qt(e,t),a=[],r=!1;return te(t,(function(t){t.inYRange(i.y)&&a.push(t),t.inRange(i.x,i.y)&&(r=!0)})),n.intersect&&!r&&(a=[]),a}}},oe=H.extend;function se(t,e){return H.where(t,(function(t){return t.pos===e}))}function le(t,e){return t.sort((function(t,n){var i=e?n:t,a=e?t:n;return i.weight===a.weight?i.index-a.index:i.weight-a.weight}))}function ue(t,e,n,i){return Math.max(t[n],e[n])+Math.max(t[i],e[i])}function de(t,e,n){var i,a,r=n.box,o=t.maxPadding;if(n.size&&(t[n.pos]-=n.size),n.size=n.horizontal?r.height:r.width,t[n.pos]+=n.size,r.getPadding){var s=r.getPadding();o.top=Math.max(o.top,s.top),o.left=Math.max(o.left,s.left),o.bottom=Math.max(o.bottom,s.bottom),o.right=Math.max(o.right,s.right)}if(i=e.outerWidth-ue(o,t,"left","right"),a=e.outerHeight-ue(o,t,"top","bottom"),i!==t.w||a!==t.h)return t.w=i,t.h=a,n.horizontal?i!==t.w:a!==t.h}function he(t,e){var n=e.maxPadding;function i(t){var i={left:0,top:0,right:0,bottom:0};return t.forEach((function(t){i[t]=Math.max(e[t],n[t])})),i}return i(t?["left","right"]:["top","bottom"])}function ce(t,e,n){var i,a,r,o,s,l,u=[];for(i=0,a=t.length;idiv{position:absolute;width:1000000px;height:1000000px;left:0;top:0}.chartjs-size-monitor-shrink>div{position:absolute;width:200%;height:200%;left:0;top:0}"}))&&ge.default||ge,ve="$chartjs",be="chartjs-size-monitor",ye="chartjs-render-monitor",xe="chartjs-render-animation",_e=["animationstart","webkitAnimationStart"],we={touchstart:"mousedown",touchmove:"mousemove",touchend:"mouseup",pointerenter:"mouseenter",pointerdown:"mousedown",pointermove:"mousemove",pointerup:"mouseup",pointerleave:"mouseout",pointerout:"mouseout"};function ke(t,e){var n=H.getStyle(t,e),i=n&&n.match(/^(\d+)(\.\d+)?px$/);return i?Number(i[1]):void 0}var Me=!!function(){var t=!1;try{var e=Object.defineProperty({},"passive",{get:function(){t=!0}});window.addEventListener("e",null,e)}catch(t){}return t}()&&{passive:!0};function Se(t,e,n){t.addEventListener(e,n,Me)}function De(t,e,n){t.removeEventListener(e,n,Me)}function Ce(t,e,n,i,a){return{type:t,chart:e,native:a||null,x:void 0!==n?n:null,y:void 0!==i?i:null}}function Pe(t){var e=document.createElement("div");return e.className=t||"",e}function Te(t,e,n){var i,a,r,o,s=t[ve]||(t[ve]={}),l=s.resizer=function(t){var e=Pe(be),n=Pe(be+"-expand"),i=Pe(be+"-shrink");n.appendChild(Pe()),i.appendChild(Pe()),e.appendChild(n),e.appendChild(i),e._reset=function(){n.scrollLeft=1e6,n.scrollTop=1e6,i.scrollLeft=1e6,i.scrollTop=1e6};var a=function(){e._reset(),t()};return Se(n,"scroll",a.bind(n,"expand")),Se(i,"scroll",a.bind(i,"shrink")),e}((i=function(){if(s.resizer){var i=n.options.maintainAspectRatio&&t.parentNode,a=i?i.clientWidth:0;e(Ce("resize",n)),i&&i.clientWidth0){var r=t[0];r.label?n=r.label:r.xLabel?n=r.xLabel:a>0&&r.index-1?t.split("\n"):t}function Ve(t){var e=W.global;return{xPadding:t.xPadding,yPadding:t.yPadding,xAlign:t.xAlign,yAlign:t.yAlign,rtl:t.rtl,textDirection:t.textDirection,bodyFontColor:t.bodyFontColor,_bodyFontFamily:Ne(t.bodyFontFamily,e.defaultFontFamily),_bodyFontStyle:Ne(t.bodyFontStyle,e.defaultFontStyle),_bodyAlign:t.bodyAlign,bodyFontSize:Ne(t.bodyFontSize,e.defaultFontSize),bodySpacing:t.bodySpacing,titleFontColor:t.titleFontColor,_titleFontFamily:Ne(t.titleFontFamily,e.defaultFontFamily),_titleFontStyle:Ne(t.titleFontStyle,e.defaultFontStyle),titleFontSize:Ne(t.titleFontSize,e.defaultFontSize),_titleAlign:t.titleAlign,titleSpacing:t.titleSpacing,titleMarginBottom:t.titleMarginBottom,footerFontColor:t.footerFontColor,_footerFontFamily:Ne(t.footerFontFamily,e.defaultFontFamily),_footerFontStyle:Ne(t.footerFontStyle,e.defaultFontStyle),footerFontSize:Ne(t.footerFontSize,e.defaultFontSize),_footerAlign:t.footerAlign,footerSpacing:t.footerSpacing,footerMarginTop:t.footerMarginTop,caretSize:t.caretSize,cornerRadius:t.cornerRadius,backgroundColor:t.backgroundColor,opacity:0,legendColorBackground:t.multiKeyBackground,displayColors:t.displayColors,borderColor:t.borderColor,borderWidth:t.borderWidth}}function He(t,e){return"center"===e?t.x+t.width/2:"right"===e?t.x+t.width-t.xPadding:t.x+t.xPadding}function Be(t){return ze([],Ee(t))}var je=$.extend({initialize:function(){this._model=Ve(this._options),this._lastActive=[]},getTitle:function(){var t=this,e=t._options,n=e.callbacks,i=n.beforeTitle.apply(t,arguments),a=n.title.apply(t,arguments),r=n.afterTitle.apply(t,arguments),o=[];return o=ze(o,Ee(i)),o=ze(o,Ee(a)),o=ze(o,Ee(r))},getBeforeBody:function(){return Be(this._options.callbacks.beforeBody.apply(this,arguments))},getBody:function(t,e){var n=this,i=n._options.callbacks,a=[];return H.each(t,(function(t){var r={before:[],lines:[],after:[]};ze(r.before,Ee(i.beforeLabel.call(n,t,e))),ze(r.lines,i.label.call(n,t,e)),ze(r.after,Ee(i.afterLabel.call(n,t,e))),a.push(r)})),a},getAfterBody:function(){return Be(this._options.callbacks.afterBody.apply(this,arguments))},getFooter:function(){var t=this,e=t._options.callbacks,n=e.beforeFooter.apply(t,arguments),i=e.footer.apply(t,arguments),a=e.afterFooter.apply(t,arguments),r=[];return r=ze(r,Ee(n)),r=ze(r,Ee(i)),r=ze(r,Ee(a))},update:function(t){var e,n,i,a,r,o,s,l,u,d,h=this,c=h._options,f=h._model,g=h._model=Ve(c),m=h._active,p=h._data,v={xAlign:f.xAlign,yAlign:f.yAlign},b={x:f.x,y:f.y},y={width:f.width,height:f.height},x={x:f.caretX,y:f.caretY};if(m.length){g.opacity=1;var _=[],w=[];x=Ye[c.position].call(h,m,h._eventPosition);var k=[];for(e=0,n=m.length;ei.width&&(a=i.width-e.width),a<0&&(a=0)),"top"===d?r+=h:r-="bottom"===d?e.height+h:e.height/2,"center"===d?"left"===u?a+=h:"right"===u&&(a-=h):"left"===u?a-=c:"right"===u&&(a+=c),{x:a,y:r}}(g,y,v=function(t,e){var n,i,a,r,o,s=t._model,l=t._chart,u=t._chart.chartArea,d="center",h="center";s.yl.height-e.height&&(h="bottom");var c=(u.left+u.right)/2,f=(u.top+u.bottom)/2;"center"===h?(n=function(t){return t<=c},i=function(t){return t>c}):(n=function(t){return t<=e.width/2},i=function(t){return t>=l.width-e.width/2}),a=function(t){return t+e.width+s.caretSize+s.caretPadding>l.width},r=function(t){return t-e.width-s.caretSize-s.caretPadding<0},o=function(t){return t<=f?"top":"bottom"},n(s.x)?(d="left",a(s.x)&&(d="center",h=o(s.y))):i(s.x)&&(d="right",r(s.x)&&(d="center",h=o(s.y)));var g=t._options;return{xAlign:g.xAlign?g.xAlign:d,yAlign:g.yAlign?g.yAlign:h}}(this,y),h._chart)}else g.opacity=0;return g.xAlign=v.xAlign,g.yAlign=v.yAlign,g.x=b.x,g.y=b.y,g.width=y.width,g.height=y.height,g.caretX=x.x,g.caretY=x.y,h._model=g,t&&c.custom&&c.custom.call(h,g),h},drawCaret:function(t,e){var n=this._chart.ctx,i=this._view,a=this.getCaretPosition(t,e,i);n.lineTo(a.x1,a.y1),n.lineTo(a.x2,a.y2),n.lineTo(a.x3,a.y3)},getCaretPosition:function(t,e,n){var i,a,r,o,s,l,u=n.caretSize,d=n.cornerRadius,h=n.xAlign,c=n.yAlign,f=t.x,g=t.y,m=e.width,p=e.height;if("center"===c)s=g+p/2,"left"===h?(a=(i=f)-u,r=i,o=s+u,l=s-u):(a=(i=f+m)+u,r=i,o=s-u,l=s+u);else if("left"===h?(i=(a=f+d+u)-u,r=a+u):"right"===h?(i=(a=f+m-d-u)-u,r=a+u):(i=(a=n.caretX)-u,r=a+u),"top"===c)s=(o=g)-u,l=o;else{s=(o=g+p)+u,l=o;var v=r;r=i,i=v}return{x1:i,x2:a,x3:r,y1:o,y2:s,y3:l}},drawTitle:function(t,e,n){var i,a,r,o=e.title,s=o.length;if(s){var l=We(e.rtl,e.x,e.width);for(t.x=He(e,e._titleAlign),n.textAlign=l.textAlign(e._titleAlign),n.textBaseline="middle",i=e.titleFontSize,a=e.titleSpacing,n.fillStyle=e.titleFontColor,n.font=H.fontString(i,e._titleFontStyle,e._titleFontFamily),r=0;r0&&n.stroke()},draw:function(){var t=this._chart.ctx,e=this._view;if(0!==e.opacity){var n={width:e.width,height:e.height},i={x:e.x,y:e.y},a=Math.abs(e.opacity<.001)?0:e.opacity,r=e.title.length||e.beforeBody.length||e.body.length||e.afterBody.length||e.footer.length;this._options.enabled&&r&&(t.save(),t.globalAlpha=a,this.drawBackground(i,e,t,n),i.y+=e.yPadding,H.rtl.overrideTextDirection(t,e.textDirection),this.drawTitle(i,e,t),this.drawBody(i,e,t),this.drawFooter(i,e,t),H.rtl.restoreTextDirection(t,e.textDirection),t.restore())}},handleEvent:function(t){var e,n=this,i=n._options;return n._lastActive=n._lastActive||[],"mouseout"===t.type?n._active=[]:(n._active=n._chart.getElementsAtEventForMode(t,i.mode,i),i.reverse&&n._active.reverse()),(e=!H.arrayEquals(n._active,n._lastActive))&&(n._lastActive=n._active,(i.enabled||i.custom)&&(n._eventPosition={x:t.x,y:t.y},n.update(!0),n.pivot())),e}}),Ue=Ye,Ge=je;Ge.positioners=Ue;var qe=H.valueOrDefault;function Ze(){return H.merge({},[].slice.call(arguments),{merger:function(t,e,n,i){if("xAxes"===t||"yAxes"===t){var a,r,o,s=n[t].length;for(e[t]||(e[t]=[]),a=0;a=e[t].length&&e[t].push({}),!e[t][a].type||o.type&&o.type!==e[t][a].type?H.merge(e[t][a],[Re.getScaleDefaults(r),o]):H.merge(e[t][a],o)}else H._merger(t,e,n,i)}})}function $e(){return H.merge({},[].slice.call(arguments),{merger:function(t,e,n,i){var a=e[t]||{},r=n[t];"scales"===t?e[t]=Ze(a,r):"scale"===t?e[t]=H.merge(a,[Re.getScaleDefaults(r.type),r]):H._merger(t,e,n,i)}})}function Xe(t){var e=t.options;H.each(t.scales,(function(e){me.removeBox(t,e)})),e=$e(W.global,W[t.config.type],e),t.options=t.config.options=e,t.ensureScalesHaveIDs(),t.buildOrUpdateScales(),t.tooltip._options=e.tooltips,t.tooltip.initialize()}function Ke(t,e,n){var i,a=function(t){return t.id===i};do{i=e+n++}while(H.findIndex(t,a)>=0);return i}function Je(t){return"top"===t||"bottom"===t}function Qe(t,e){return function(n,i){return n[t]===i[t]?n[e]-i[e]:n[t]-i[t]}}W._set("global",{elements:{},events:["mousemove","mouseout","click","touchstart","touchmove"],hover:{onHover:null,mode:"nearest",intersect:!0,animationDuration:400},onClick:null,maintainAspectRatio:!0,responsive:!0,responsiveAnimationDuration:0});var tn=function(t,e){return this.construct(t,e),this};H.extend(tn.prototype,{construct:function(t,e){var n=this;e=function(t){var e=(t=t||{}).data=t.data||{};return e.datasets=e.datasets||[],e.labels=e.labels||[],t.options=$e(W.global,W[t.type],t.options||{}),t}(e);var i=Ie.acquireContext(t,e),a=i&&i.canvas,r=a&&a.height,o=a&&a.width;n.id=H.uid(),n.ctx=i,n.canvas=a,n.config=e,n.width=o,n.height=r,n.aspectRatio=r?o/r:null,n.options=e.options,n._bufferedRender=!1,n._layers=[],n.chart=n,n.controller=n,tn.instances[n.id]=n,Object.defineProperty(n,"data",{get:function(){return n.config.data},set:function(t){n.config.data=t}}),i&&a?(n.initialize(),n.update()):console.error("Failed to create chart: can't acquire context from the given item")},initialize:function(){var t=this;return Le.notify(t,"beforeInit"),H.retinaScale(t,t.options.devicePixelRatio),t.bindEvents(),t.options.responsive&&t.resize(!0),t.initToolTip(),Le.notify(t,"afterInit"),t},clear:function(){return H.canvas.clear(this),this},stop:function(){return J.cancelAnimation(this),this},resize:function(t){var e=this,n=e.options,i=e.canvas,a=n.maintainAspectRatio&&e.aspectRatio||null,r=Math.max(0,Math.floor(H.getMaximumWidth(i))),o=Math.max(0,Math.floor(a?r/a:H.getMaximumHeight(i)));if((e.width!==r||e.height!==o)&&(i.width=e.width=r,i.height=e.height=o,i.style.width=r+"px",i.style.height=o+"px",H.retinaScale(e,n.devicePixelRatio),!t)){var s={width:r,height:o};Le.notify(e,"resize",[s]),n.onResize&&n.onResize(e,s),e.stop(),e.update({duration:n.responsiveAnimationDuration})}},ensureScalesHaveIDs:function(){var t=this.options,e=t.scales||{},n=t.scale;H.each(e.xAxes,(function(t,n){t.id||(t.id=Ke(e.xAxes,"x-axis-",n))})),H.each(e.yAxes,(function(t,n){t.id||(t.id=Ke(e.yAxes,"y-axis-",n))})),n&&(n.id=n.id||"scale")},buildOrUpdateScales:function(){var t=this,e=t.options,n=t.scales||{},i=[],a=Object.keys(n).reduce((function(t,e){return t[e]=!1,t}),{});e.scales&&(i=i.concat((e.scales.xAxes||[]).map((function(t){return{options:t,dtype:"category",dposition:"bottom"}})),(e.scales.yAxes||[]).map((function(t){return{options:t,dtype:"linear",dposition:"left"}})))),e.scale&&i.push({options:e.scale,dtype:"radialLinear",isDefault:!0,dposition:"chartArea"}),H.each(i,(function(e){var i=e.options,r=i.id,o=qe(i.type,e.dtype);Je(i.position)!==Je(e.dposition)&&(i.position=e.dposition),a[r]=!0;var s=null;if(r in n&&n[r].type===o)(s=n[r]).options=i,s.ctx=t.ctx,s.chart=t;else{var l=Re.getScaleConstructor(o);if(!l)return;s=new l({id:r,type:o,options:i,ctx:t.ctx,chart:t}),n[s.id]=s}s.mergeTicksOptions(),e.isDefault&&(t.scale=s)})),H.each(a,(function(t,e){t||delete n[e]})),t.scales=n,Re.addScalesToLayout(this)},buildOrUpdateControllers:function(){var t,e,n=this,i=[],a=n.data.datasets;for(t=0,e=a.length;t=0;--n)this.drawDataset(e[n],t);Le.notify(this,"afterDatasetsDraw",[t])}},drawDataset:function(t,e){var n={meta:t,index:t.index,easingValue:e};!1!==Le.notify(this,"beforeDatasetDraw",[n])&&(t.controller.draw(e),Le.notify(this,"afterDatasetDraw",[n]))},_drawTooltip:function(t){var e=this.tooltip,n={tooltip:e,easingValue:t};!1!==Le.notify(this,"beforeTooltipDraw",[n])&&(e.draw(),Le.notify(this,"afterTooltipDraw",[n]))},getElementAtEvent:function(t){return re.modes.single(this,t)},getElementsAtEvent:function(t){return re.modes.label(this,t,{intersect:!0})},getElementsAtXAxis:function(t){return re.modes["x-axis"](this,t,{intersect:!0})},getElementsAtEventForMode:function(t,e,n){var i=re.modes[e];return"function"==typeof i?i(this,t,n):[]},getDatasetAtEvent:function(t){return re.modes.dataset(this,t,{intersect:!0})},getDatasetMeta:function(t){var e=this.data.datasets[t];e._meta||(e._meta={});var n=e._meta[this.id];return n||(n=e._meta[this.id]={type:null,data:[],dataset:null,controller:null,hidden:null,xAxisID:null,yAxisID:null,order:e.order||0,index:t}),n},getVisibleDatasetCount:function(){for(var t=0,e=0,n=this.data.datasets.length;e3?n[2]-n[1]:n[1]-n[0];Math.abs(i)>1&&t!==Math.floor(t)&&(i=t-Math.floor(t));var a=H.log10(Math.abs(i)),r="";if(0!==t)if(Math.max(Math.abs(n[0]),Math.abs(n[n.length-1]))<1e-4){var o=H.log10(Math.abs(t)),s=Math.floor(o)-Math.floor(a);s=Math.max(Math.min(s,20),0),r=t.toExponential(s)}else{var l=-1*Math.floor(a);l=Math.max(Math.min(l,20),0),r=t.toFixed(l)}else r="0";return r},logarithmic:function(t,e,n){var i=t/Math.pow(10,Math.floor(H.log10(t)));return 0===t?"0":1===i||2===i||5===i||0===e||e===n.length-1?t.toExponential():""}}},sn=H.isArray,ln=H.isNullOrUndef,un=H.valueOrDefault,dn=H.valueAtIndexOrDefault;function hn(t,e,n){var i,a=t.getTicks().length,r=Math.min(e,a-1),o=t.getPixelForTick(r),s=t._startPixel,l=t._endPixel;if(!(n&&(i=1===a?Math.max(o-s,l-o):0===e?(t.getPixelForTick(1)-o)/2:(o-t.getPixelForTick(r-1))/2,(o+=rl+1e-6)))return o}function cn(t,e,n,i){var a,r,o,s,l,u,d,h,c,f,g,m,p,v=n.length,b=[],y=[],x=[];for(a=0;ae){for(n=0;n=c||d<=1||!s.isHorizontal()?s.labelRotation=h:(e=(t=s._getLabelSizes()).widest.width,n=t.highest.height-t.highest.offset,i=Math.min(s.maxWidth,s.chart.width-e),e+6>(a=l.offset?s.maxWidth/d:i/(d-1))&&(a=i/(d-(l.offset?.5:1)),r=s.maxHeight-fn(l.gridLines)-u.padding-gn(l.scaleLabel),o=Math.sqrt(e*e+n*n),f=H.toDegrees(Math.min(Math.asin(Math.min((t.highest.height+6)/a,1)),Math.asin(Math.min(r/o,1))-Math.asin(n/o))),f=Math.max(h,Math.min(c,f))),s.labelRotation=f)},afterCalculateTickRotation:function(){H.callback(this.options.afterCalculateTickRotation,[this])},beforeFit:function(){H.callback(this.options.beforeFit,[this])},fit:function(){var t=this,e=t.minSize={width:0,height:0},n=t.chart,i=t.options,a=i.ticks,r=i.scaleLabel,o=i.gridLines,s=t._isVisible(),l="bottom"===i.position,u=t.isHorizontal();if(u?e.width=t.maxWidth:s&&(e.width=fn(o)+gn(r)),u?s&&(e.height=fn(o)+gn(r)):e.height=t.maxHeight,a.display&&s){var d=pn(a),h=t._getLabelSizes(),c=h.first,f=h.last,g=h.widest,m=h.highest,p=.4*d.minor.lineHeight,v=a.padding;if(u){var b=0!==t.labelRotation,y=H.toRadians(t.labelRotation),x=Math.cos(y),_=Math.sin(y),w=_*g.width+x*(m.height-(b?m.offset:0))+(b?0:p);e.height=Math.min(t.maxHeight,e.height+w+v);var k,M,S=t.getPixelForTick(0)-t.left,D=t.right-t.getPixelForTick(t.getTicks().length-1);b?(k=l?x*c.width+_*c.offset:_*(c.height-c.offset),M=l?_*(f.height-f.offset):x*f.width+_*f.offset):(k=c.width/2,M=f.width/2),t.paddingLeft=Math.max((k-S)*t.width/(t.width-S),0)+3,t.paddingRight=Math.max((M-D)*t.width/(t.width-D),0)+3}else{var C=a.mirror?0:g.width+v+p;e.width=Math.min(t.maxWidth,e.width+C),t.paddingTop=c.height/2,t.paddingBottom=f.height/2}}t.handleMargins(),u?(t.width=t._length=n.width-t.margins.left-t.margins.right,t.height=e.height):(t.width=e.width,t.height=t._length=n.height-t.margins.top-t.margins.bottom)},handleMargins:function(){var t=this;t.margins&&(t.margins.left=Math.max(t.paddingLeft,t.margins.left),t.margins.top=Math.max(t.paddingTop,t.margins.top),t.margins.right=Math.max(t.paddingRight,t.margins.right),t.margins.bottom=Math.max(t.paddingBottom,t.margins.bottom))},afterFit:function(){H.callback(this.options.afterFit,[this])},isHorizontal:function(){var t=this.options.position;return"top"===t||"bottom"===t},isFullWidth:function(){return this.options.fullWidth},getRightValue:function(t){if(ln(t))return NaN;if(("number"==typeof t||t instanceof Number)&&!isFinite(t))return NaN;if(t)if(this.isHorizontal()){if(void 0!==t.x)return this.getRightValue(t.x)}else if(void 0!==t.y)return this.getRightValue(t.y);return t},_convertTicksToLabels:function(t){var e,n,i,a=this;for(a.ticks=t.map((function(t){return t.value})),a.beforeTickToLabelConversion(),e=a.convertTicksToLabels(t)||a.ticks,a.afterTickToLabelConversion(),n=0,i=t.length;nn-1?null:this.getPixelForDecimal(t*i+(e?i/2:0))},getPixelForDecimal:function(t){return this._reversePixels&&(t=1-t),this._startPixel+t*this._length},getDecimalForPixel:function(t){var e=(t-this._startPixel)/this._length;return this._reversePixels?1-e:e},getBasePixel:function(){return this.getPixelForValue(this.getBaseValue())},getBaseValue:function(){var t=this.min,e=this.max;return this.beginAtZero?0:t<0&&e<0?e:t>0&&e>0?t:0},_autoSkip:function(t){var e,n,i,a,r=this.options.ticks,o=this._length,s=r.maxTicksLimit||o/this._tickSize()+1,l=r.major.enabled?function(t){var e,n,i=[];for(e=0,n=t.length;es)return function(t,e,n){var i,a,r=0,o=e[0];for(n=Math.ceil(n),i=0;iu)return r;return Math.max(u,1)}(l,t,0,s),u>0){for(e=0,n=u-1;e1?(h-d)/(u-1):null,bn(t,i,H.isNullOrUndef(a)?0:d-a,d),bn(t,i,h,H.isNullOrUndef(a)?t.length:h+a),vn(t)}return bn(t,i),vn(t)},_tickSize:function(){var t=this.options.ticks,e=H.toRadians(this.labelRotation),n=Math.abs(Math.cos(e)),i=Math.abs(Math.sin(e)),a=this._getLabelSizes(),r=t.autoSkipPadding||0,o=a?a.widest.width+r:0,s=a?a.highest.height+r:0;return this.isHorizontal()?s*n>o*i?o/n:s/i:s*i=0&&(o=t),void 0!==r&&(t=n.indexOf(r))>=0&&(s=t),e.minIndex=o,e.maxIndex=s,e.min=n[o],e.max=n[s]},buildTicks:function(){var t=this._getLabels(),e=this.minIndex,n=this.maxIndex;this.ticks=0===e&&n===t.length-1?t:t.slice(e,n+1)},getLabelForIndex:function(t,e){var n=this.chart;return n.getDatasetMeta(e).controller._getValueScaleId()===this.id?this.getRightValue(n.data.datasets[e].data[t]):this._getLabels()[t]},_configure:function(){var t=this,e=t.options.offset,n=t.ticks;xn.prototype._configure.call(t),t.isHorizontal()||(t._reversePixels=!t._reversePixels),n&&(t._startValue=t.minIndex-(e?.5:0),t._valueRange=Math.max(n.length-(e?0:1),1))},getPixelForValue:function(t,e,n){var i,a,r,o=this;return _n(e)||_n(n)||(t=o.chart.data.datasets[n].data[e]),_n(t)||(i=o.isHorizontal()?t.x:t.y),(void 0!==i||void 0!==t&&isNaN(e))&&(a=o._getLabels(),t=H.valueOrDefault(i,t),e=-1!==(r=a.indexOf(t))?r:e,isNaN(e)&&(e=t)),o.getPixelForDecimal((e-o._startValue)/o._valueRange)},getPixelForTick:function(t){var e=this.ticks;return t<0||t>e.length-1?null:this.getPixelForValue(e[t],t+this.minIndex)},getValueForPixel:function(t){var e=Math.round(this._startValue+this.getDecimalForPixel(t)*this._valueRange);return Math.min(Math.max(e,0),this.ticks.length-1)},getBasePixel:function(){return this.bottom}}),kn={position:"bottom"};wn._defaults=kn;var Mn=H.noop,Sn=H.isNullOrUndef;var Dn=xn.extend({getRightValue:function(t){return"string"==typeof t?+t:xn.prototype.getRightValue.call(this,t)},handleTickRangeOptions:function(){var t=this,e=t.options.ticks;if(e.beginAtZero){var n=H.sign(t.min),i=H.sign(t.max);n<0&&i<0?t.max=0:n>0&&i>0&&(t.min=0)}var a=void 0!==e.min||void 0!==e.suggestedMin,r=void 0!==e.max||void 0!==e.suggestedMax;void 0!==e.min?t.min=e.min:void 0!==e.suggestedMin&&(null===t.min?t.min=e.suggestedMin:t.min=Math.min(t.min,e.suggestedMin)),void 0!==e.max?t.max=e.max:void 0!==e.suggestedMax&&(null===t.max?t.max=e.suggestedMax:t.max=Math.max(t.max,e.suggestedMax)),a!==r&&t.min>=t.max&&(a?t.max=t.min+1:t.min=t.max-1),t.min===t.max&&(t.max++,e.beginAtZero||t.min--)},getTickLimit:function(){var t,e=this.options.ticks,n=e.stepSize,i=e.maxTicksLimit;return n?t=Math.ceil(this.max/n)-Math.floor(this.min/n)+1:(t=this._computeTickLimit(),i=i||11),i&&(t=Math.min(i,t)),t},_computeTickLimit:function(){return Number.POSITIVE_INFINITY},handleDirectionalChanges:Mn,buildTicks:function(){var t=this,e=t.options.ticks,n=t.getTickLimit(),i={maxTicks:n=Math.max(2,n),min:e.min,max:e.max,precision:e.precision,stepSize:H.valueOrDefault(e.fixedStepSize,e.stepSize)},a=t.ticks=function(t,e){var n,i,a,r,o=[],s=t.stepSize,l=s||1,u=t.maxTicks-1,d=t.min,h=t.max,c=t.precision,f=e.min,g=e.max,m=H.niceNum((g-f)/u/l)*l;if(m<1e-14&&Sn(d)&&Sn(h))return[f,g];(r=Math.ceil(g/m)-Math.floor(f/m))>u&&(m=H.niceNum(r*m/u/l)*l),s||Sn(c)?n=Math.pow(10,H._decimalPlaces(m)):(n=Math.pow(10,c),m=Math.ceil(m*n)/n),i=Math.floor(f/m)*m,a=Math.ceil(g/m)*m,s&&(!Sn(d)&&H.almostWhole(d/m,m/1e3)&&(i=d),!Sn(h)&&H.almostWhole(h/m,m/1e3)&&(a=h)),r=(a-i)/m,r=H.almostEquals(r,Math.round(r),m/1e3)?Math.round(r):Math.ceil(r),i=Math.round(i*n)/n,a=Math.round(a*n)/n,o.push(Sn(d)?i:d);for(var p=1;pe.length-1?null:this.getPixelForValue(e[t])}}),An=Cn;On._defaults=An;var Fn=H.valueOrDefault,In=H.math.log10;var Ln={position:"left",ticks:{callback:on.formatters.logarithmic}};function Rn(t,e){return H.isFinite(t)&&t>=0?t:e}var Nn=xn.extend({determineDataLimits:function(){var t,e,n,i,a,r,o=this,s=o.options,l=o.chart,u=l.data.datasets,d=o.isHorizontal();function h(t){return d?t.xAxisID===o.id:t.yAxisID===o.id}o.min=Number.POSITIVE_INFINITY,o.max=Number.NEGATIVE_INFINITY,o.minNotZero=Number.POSITIVE_INFINITY;var c=s.stacked;if(void 0===c)for(t=0;t0){var e=H.min(t),n=H.max(t);o.min=Math.min(o.min,e),o.max=Math.max(o.max,n)}}))}else for(t=0;t0?t.minNotZero=t.min:t.max<1?t.minNotZero=Math.pow(10,Math.floor(In(t.max))):t.minNotZero=1)},buildTicks:function(){var t=this,e=t.options.ticks,n=!t.isHorizontal(),i={min:Rn(e.min),max:Rn(e.max)},a=t.ticks=function(t,e){var n,i,a=[],r=Fn(t.min,Math.pow(10,Math.floor(In(e.min)))),o=Math.floor(In(e.max)),s=Math.ceil(e.max/Math.pow(10,o));0===r?(n=Math.floor(In(e.minNotZero)),i=Math.floor(e.minNotZero/Math.pow(10,n)),a.push(r),r=i*Math.pow(10,n)):(n=Math.floor(In(r)),i=Math.floor(r/Math.pow(10,n)));var l=n<0?Math.pow(10,Math.abs(n)):1;do{a.push(r),10===++i&&(i=1,l=++n>=0?1:l),r=Math.round(i*Math.pow(10,n)*l)/l}while(ne.length-1?null:this.getPixelForValue(e[t])},_getFirstTickValue:function(t){var e=Math.floor(In(t));return Math.floor(t/Math.pow(10,e))*Math.pow(10,e)},_configure:function(){var t=this,e=t.min,n=0;xn.prototype._configure.call(t),0===e&&(e=t._getFirstTickValue(t.minNotZero),n=Fn(t.options.ticks.fontSize,W.global.defaultFontSize)/t._length),t._startValue=In(e),t._valueOffset=n,t._valueRange=(In(t.max)-In(e))/(1-n)},getPixelForValue:function(t){var e=this,n=0;return(t=+e.getRightValue(t))>e.min&&t>0&&(n=(In(t)-e._startValue)/e._valueRange+e._valueOffset),e.getPixelForDecimal(n)},getValueForPixel:function(t){var e=this,n=e.getDecimalForPixel(t);return 0===n&&0===e.min?0:Math.pow(10,e._startValue+(n-e._valueOffset)*e._valueRange)}}),Wn=Ln;Nn._defaults=Wn;var Yn=H.valueOrDefault,zn=H.valueAtIndexOrDefault,En=H.options.resolve,Vn={display:!0,animate:!0,position:"chartArea",angleLines:{display:!0,color:"rgba(0,0,0,0.1)",lineWidth:1,borderDash:[],borderDashOffset:0},gridLines:{circular:!1},ticks:{showLabelBackdrop:!0,backdropColor:"rgba(255,255,255,0.75)",backdropPaddingY:2,backdropPaddingX:2,callback:on.formatters.linear},pointLabels:{display:!0,fontSize:10,callback:function(t){return t}}};function Hn(t){var e=t.ticks;return e.display&&t.display?Yn(e.fontSize,W.global.defaultFontSize)+2*e.backdropPaddingY:0}function Bn(t,e,n,i,a){return t===i||t===a?{start:e-n/2,end:e+n/2}:ta?{start:e-n,end:e}:{start:e,end:e+n}}function jn(t){return 0===t||180===t?"center":t<180?"left":"right"}function Un(t,e,n,i){var a,r,o=n.y+i/2;if(H.isArray(e))for(a=0,r=e.length;a270||t<90)&&(n.y-=e.h)}function qn(t){return H.isNumber(t)?t:0}var Zn=Dn.extend({setDimensions:function(){var t=this;t.width=t.maxWidth,t.height=t.maxHeight,t.paddingTop=Hn(t.options)/2,t.xCenter=Math.floor(t.width/2),t.yCenter=Math.floor((t.height-t.paddingTop)/2),t.drawingArea=Math.min(t.height-t.paddingTop,t.width)/2},determineDataLimits:function(){var t=this,e=t.chart,n=Number.POSITIVE_INFINITY,i=Number.NEGATIVE_INFINITY;H.each(e.data.datasets,(function(a,r){if(e.isDatasetVisible(r)){var o=e.getDatasetMeta(r);H.each(a.data,(function(e,a){var r=+t.getRightValue(e);isNaN(r)||o.data[a].hidden||(n=Math.min(r,n),i=Math.max(r,i))}))}})),t.min=n===Number.POSITIVE_INFINITY?0:n,t.max=i===Number.NEGATIVE_INFINITY?0:i,t.handleTickRangeOptions()},_computeTickLimit:function(){return Math.ceil(this.drawingArea/Hn(this.options))},convertTicksToLabels:function(){var t=this;Dn.prototype.convertTicksToLabels.call(t),t.pointLabels=t.chart.data.labels.map((function(){var e=H.callback(t.options.pointLabels.callback,arguments,t);return e||0===e?e:""}))},getLabelForIndex:function(t,e){return+this.getRightValue(this.chart.data.datasets[e].data[t])},fit:function(){var t=this.options;t.display&&t.pointLabels.display?function(t){var e,n,i,a=H.options._parseFont(t.options.pointLabels),r={l:0,r:t.width,t:0,b:t.height-t.paddingTop},o={};t.ctx.font=a.string,t._pointLabelSizes=[];var s,l,u,d=t.chart.data.labels.length;for(e=0;er.r&&(r.r=f.end,o.r=h),g.startr.b&&(r.b=g.end,o.b=h)}t.setReductions(t.drawingArea,r,o)}(this):this.setCenterPoint(0,0,0,0)},setReductions:function(t,e,n){var i=this,a=e.l/Math.sin(n.l),r=Math.max(e.r-i.width,0)/Math.sin(n.r),o=-e.t/Math.cos(n.t),s=-Math.max(e.b-(i.height-i.paddingTop),0)/Math.cos(n.b);a=qn(a),r=qn(r),o=qn(o),s=qn(s),i.drawingArea=Math.min(Math.floor(t-(a+r)/2),Math.floor(t-(o+s)/2)),i.setCenterPoint(a,r,o,s)},setCenterPoint:function(t,e,n,i){var a=this,r=a.width-e-a.drawingArea,o=t+a.drawingArea,s=n+a.drawingArea,l=a.height-a.paddingTop-i-a.drawingArea;a.xCenter=Math.floor((o+r)/2+a.left),a.yCenter=Math.floor((s+l)/2+a.top+a.paddingTop)},getIndexAngle:function(t){var e=this.chart,n=(t*(360/e.data.labels.length)+((e.options||{}).startAngle||0))%360;return(n<0?n+360:n)*Math.PI*2/360},getDistanceFromCenterForValue:function(t){var e=this;if(H.isNullOrUndef(t))return NaN;var n=e.drawingArea/(e.max-e.min);return e.options.ticks.reverse?(e.max-t)*n:(t-e.min)*n},getPointPosition:function(t,e){var n=this.getIndexAngle(t)-Math.PI/2;return{x:Math.cos(n)*e+this.xCenter,y:Math.sin(n)*e+this.yCenter}},getPointPositionForValue:function(t,e){return this.getPointPosition(t,this.getDistanceFromCenterForValue(e))},getBasePosition:function(t){var e=this.min,n=this.max;return this.getPointPositionForValue(t||0,this.beginAtZero?0:e<0&&n<0?n:e>0&&n>0?e:0)},_drawGrid:function(){var t,e,n,i=this,a=i.ctx,r=i.options,o=r.gridLines,s=r.angleLines,l=Yn(s.lineWidth,o.lineWidth),u=Yn(s.color,o.color);if(r.pointLabels.display&&function(t){var e=t.ctx,n=t.options,i=n.pointLabels,a=Hn(n),r=t.getDistanceFromCenterForValue(n.ticks.reverse?t.min:t.max),o=H.options._parseFont(i);e.save(),e.font=o.string,e.textBaseline="middle";for(var s=t.chart.data.labels.length-1;s>=0;s--){var l=0===s?a/2:0,u=t.getPointPosition(s,r+l+5),d=zn(i.fontColor,s,W.global.defaultFontColor);e.fillStyle=d;var h=t.getIndexAngle(s),c=H.toDegrees(h);e.textAlign=jn(c),Gn(c,t._pointLabelSizes[s],u),Un(e,t.pointLabels[s],u,o.lineHeight)}e.restore()}(i),o.display&&H.each(i.ticks,(function(t,n){0!==n&&(e=i.getDistanceFromCenterForValue(i.ticksAsNumbers[n]),function(t,e,n,i){var a,r=t.ctx,o=e.circular,s=t.chart.data.labels.length,l=zn(e.color,i-1),u=zn(e.lineWidth,i-1);if((o||s)&&l&&u){if(r.save(),r.strokeStyle=l,r.lineWidth=u,r.setLineDash&&(r.setLineDash(e.borderDash||[]),r.lineDashOffset=e.borderDashOffset||0),r.beginPath(),o)r.arc(t.xCenter,t.yCenter,n,0,2*Math.PI);else{a=t.getPointPosition(0,n),r.moveTo(a.x,a.y);for(var d=1;d=0;t--)e=i.getDistanceFromCenterForValue(r.ticks.reverse?i.min:i.max),n=i.getPointPosition(t,e),a.beginPath(),a.moveTo(i.xCenter,i.yCenter),a.lineTo(n.x,n.y),a.stroke();a.restore()}},_drawLabels:function(){var t=this,e=t.ctx,n=t.options.ticks;if(n.display){var i,a,r=t.getIndexAngle(0),o=H.options._parseFont(n),s=Yn(n.fontColor,W.global.defaultFontColor);e.save(),e.font=o.string,e.translate(t.xCenter,t.yCenter),e.rotate(r),e.textAlign="center",e.textBaseline="middle",H.each(t.ticks,(function(r,l){(0!==l||n.reverse)&&(i=t.getDistanceFromCenterForValue(t.ticksAsNumbers[l]),n.showLabelBackdrop&&(a=e.measureText(r).width,e.fillStyle=n.backdropColor,e.fillRect(-a/2-n.backdropPaddingX,-i-o.size/2-n.backdropPaddingY,a+2*n.backdropPaddingX,o.size+2*n.backdropPaddingY)),e.fillStyle=s,e.fillText(r,0,-i))})),e.restore()}},_drawTitle:H.noop}),$n=Vn;Zn._defaults=$n;var Xn=H._deprecated,Kn=H.options.resolve,Jn=H.valueOrDefault,Qn=Number.MIN_SAFE_INTEGER||-9007199254740991,ti=Number.MAX_SAFE_INTEGER||9007199254740991,ei={millisecond:{common:!0,size:1,steps:1e3},second:{common:!0,size:1e3,steps:60},minute:{common:!0,size:6e4,steps:60},hour:{common:!0,size:36e5,steps:24},day:{common:!0,size:864e5,steps:30},week:{common:!1,size:6048e5,steps:4},month:{common:!0,size:2628e6,steps:12},quarter:{common:!1,size:7884e6,steps:4},year:{common:!0,size:3154e7}},ni=Object.keys(ei);function ii(t,e){return t-e}function ai(t){return H.valueOrDefault(t.time.min,t.ticks.min)}function ri(t){return H.valueOrDefault(t.time.max,t.ticks.max)}function oi(t,e,n,i){var a=function(t,e,n){for(var i,a,r,o=0,s=t.length-1;o>=0&&o<=s;){if(a=t[(i=o+s>>1)-1]||null,r=t[i],!a)return{lo:null,hi:r};if(r[e]n))return{lo:a,hi:r};s=i-1}}return{lo:r,hi:null}}(t,e,n),r=a.lo?a.hi?a.lo:t[t.length-2]:t[0],o=a.lo?a.hi?a.hi:t[t.length-1]:t[1],s=o[e]-r[e],l=s?(n-r[e])/s:0,u=(o[i]-r[i])*l;return r[i]+u}function si(t,e){var n=t._adapter,i=t.options.time,a=i.parser,r=a||i.format,o=e;return"function"==typeof a&&(o=a(o)),H.isFinite(o)||(o="string"==typeof r?n.parse(o,r):n.parse(o)),null!==o?+o:(a||"function"!=typeof r||(o=r(e),H.isFinite(o)||(o=n.parse(o))),o)}function li(t,e){if(H.isNullOrUndef(e))return null;var n=t.options.time,i=si(t,t.getRightValue(e));return null===i?i:(n.round&&(i=+t._adapter.startOf(i,n.round)),i)}function ui(t,e,n,i){var a,r,o,s=ni.length;for(a=ni.indexOf(t);a=0&&(e[r].major=!0);return e}(t,r,o,n):r}var hi=xn.extend({initialize:function(){this.mergeTicksOptions(),xn.prototype.initialize.call(this)},update:function(){var t=this,e=t.options,n=e.time||(e.time={}),i=t._adapter=new rn._date(e.adapters.date);return Xn("time scale",n.format,"time.format","time.parser"),Xn("time scale",n.min,"time.min","ticks.min"),Xn("time scale",n.max,"time.max","ticks.max"),H.mergeIf(n.displayFormats,i.formats()),xn.prototype.update.apply(t,arguments)},getRightValue:function(t){return t&&void 0!==t.t&&(t=t.t),xn.prototype.getRightValue.call(this,t)},determineDataLimits:function(){var t,e,n,i,a,r,o,s=this,l=s.chart,u=s._adapter,d=s.options,h=d.time.unit||"day",c=ti,f=Qn,g=[],m=[],p=[],v=s._getLabels();for(t=0,n=v.length;t1?function(t){var e,n,i,a={},r=[];for(e=0,n=t.length;e1e5*u)throw e+" and "+n+" are too far apart with stepSize of "+u+" "+l;for(a=h;a=a&&n<=r&&d.push(n);return i.min=a,i.max=r,i._unit=l.unit||(s.autoSkip?ui(l.minUnit,i.min,i.max,h):function(t,e,n,i,a){var r,o;for(r=ni.length-1;r>=ni.indexOf(n);r--)if(o=ni[r],ei[o].common&&t._adapter.diff(a,i,o)>=e-1)return o;return ni[n?ni.indexOf(n):0]}(i,d.length,l.minUnit,i.min,i.max)),i._majorUnit=s.major.enabled&&"year"!==i._unit?function(t){for(var e=ni.indexOf(t)+1,n=ni.length;ee&&s=0&&t0?s:1}}),ci={position:"bottom",distribution:"linear",bounds:"data",adapters:{},time:{parser:!1,unit:!1,round:!1,displayFormat:!1,isoWeekday:!1,minUnit:"millisecond",displayFormats:{}},ticks:{autoSkip:!1,source:"auto",major:{enabled:!1}}};hi._defaults=ci;var fi={category:wn,linear:On,logarithmic:Nn,radialLinear:Zn,time:hi},gi=e((function(e,n){e.exports=function(){var n,i;function a(){return n.apply(null,arguments)}function r(t){return t instanceof Array||"[object Array]"===Object.prototype.toString.call(t)}function o(t){return null!=t&&"[object Object]"===Object.prototype.toString.call(t)}function s(t){return void 0===t}function l(t){return"number"==typeof t||"[object Number]"===Object.prototype.toString.call(t)}function u(t){return t instanceof Date||"[object Date]"===Object.prototype.toString.call(t)}function d(t,e){var n,i=[];for(n=0;n>>0,i=0;i0)for(n=0;n=0?n?"+":"":"-")+Math.pow(10,Math.max(0,a)).toString().substr(1)+i}var E=/(\[[^\[]*\])|(\\)?([Hh]mm(ss)?|Mo|MM?M?M?|Do|DDDo|DD?D?D?|ddd?d?|do?|w[o|w]?|W[o|W]?|Qo?|YYYYYY|YYYYY|YYYY|YY|gg(ggg?)?|GG(GGG?)?|e|E|a|A|hh?|HH?|kk?|mm?|ss?|S{1,9}|x|X|zz?|ZZ?|.)/g,V=/(\[[^\[]*\])|(\\)?(LTS|LT|LL?L?L?|l{1,4})/g,H={},B={};function j(t,e,n,i){var a=i;"string"==typeof i&&(a=function(){return this[i]()}),t&&(B[t]=a),e&&(B[e[0]]=function(){return z(a.apply(this,arguments),e[1],e[2])}),n&&(B[n]=function(){return this.localeData().ordinal(a.apply(this,arguments),t)})}function U(t,e){return t.isValid()?(e=G(e,t.localeData()),H[e]=H[e]||function(t){var e,n,i,a=t.match(E);for(e=0,n=a.length;e=0&&V.test(t);)t=t.replace(V,i),V.lastIndex=0,n-=1;return t}var q=/\d/,Z=/\d\d/,$=/\d{3}/,X=/\d{4}/,K=/[+-]?\d{6}/,J=/\d\d?/,Q=/\d\d\d\d?/,tt=/\d\d\d\d\d\d?/,et=/\d{1,3}/,nt=/\d{1,4}/,it=/[+-]?\d{1,6}/,at=/\d+/,rt=/[+-]?\d+/,ot=/Z|[+-]\d\d:?\d\d/gi,st=/Z|[+-]\d\d(?::?\d\d)?/gi,lt=/[0-9]{0,256}['a-z\u00A0-\u05FF\u0700-\uD7FF\uF900-\uFDCF\uFDF0-\uFF07\uFF10-\uFFEF]{1,256}|[\u0600-\u06FF\/]{1,256}(\s*?[\u0600-\u06FF]{1,256}){1,2}/i,ut={};function dt(t,e,n){ut[t]=O(e)?e:function(t,i){return t&&n?n:e}}function ht(t,e){return h(ut,t)?ut[t](e._strict,e._locale):new RegExp(ct(t.replace("\\","").replace(/\\(\[)|\\(\])|\[([^\]\[]*)\]|\\(.)/g,(function(t,e,n,i,a){return e||n||i||a}))))}function ct(t){return t.replace(/[-\/\\^$*+?.()|[\]{}]/g,"\\$&")}var ft={};function gt(t,e){var n,i=e;for("string"==typeof t&&(t=[t]),l(e)&&(i=function(t,n){n[e]=k(t)}),n=0;n68?1900:2e3)};var Pt,Tt=Ot("FullYear",!0);function Ot(t,e){return function(n){return null!=n?(Ft(this,t,n),a.updateOffset(this,e),this):At(this,t)}}function At(t,e){return t.isValid()?t._d["get"+(t._isUTC?"UTC":"")+e]():NaN}function Ft(t,e,n){t.isValid()&&!isNaN(n)&&("FullYear"===e&&Ct(t.year())&&1===t.month()&&29===t.date()?t._d["set"+(t._isUTC?"UTC":"")+e](n,t.month(),It(n,t.month())):t._d["set"+(t._isUTC?"UTC":"")+e](n))}function It(t,e){if(isNaN(t)||isNaN(e))return NaN;var n=function(t,e){return(t%e+e)%e}(e,12);return t+=(e-n)/12,1===n?Ct(t)?29:28:31-n%7%2}Pt=Array.prototype.indexOf?Array.prototype.indexOf:function(t){var e;for(e=0;e=0?(s=new Date(t+400,e,n,i,a,r,o),isFinite(s.getFullYear())&&s.setFullYear(t)):s=new Date(t,e,n,i,a,r,o),s}function jt(t){var e;if(t<100&&t>=0){var n=Array.prototype.slice.call(arguments);n[0]=t+400,e=new Date(Date.UTC.apply(null,n)),isFinite(e.getUTCFullYear())&&e.setUTCFullYear(t)}else e=new Date(Date.UTC.apply(null,arguments));return e}function Ut(t,e,n){var i=7+e-n;return-(7+jt(t,0,i).getUTCDay()-e)%7+i-1}function Gt(t,e,n,i,a){var r,o,s=1+7*(e-1)+(7+n-i)%7+Ut(t,i,a);return s<=0?o=Dt(r=t-1)+s:s>Dt(t)?(r=t+1,o=s-Dt(t)):(r=t,o=s),{year:r,dayOfYear:o}}function qt(t,e,n){var i,a,r=Ut(t.year(),e,n),o=Math.floor((t.dayOfYear()-r-1)/7)+1;return o<1?i=o+Zt(a=t.year()-1,e,n):o>Zt(t.year(),e,n)?(i=o-Zt(t.year(),e,n),a=t.year()+1):(a=t.year(),i=o),{week:i,year:a}}function Zt(t,e,n){var i=Ut(t,e,n),a=Ut(t+1,e,n);return(Dt(t)-i+a)/7}function $t(t,e){return t.slice(e,7).concat(t.slice(0,e))}j("w",["ww",2],"wo","week"),j("W",["WW",2],"Wo","isoWeek"),L("week","w"),L("isoWeek","W"),Y("week",5),Y("isoWeek",5),dt("w",J),dt("ww",J,Z),dt("W",J),dt("WW",J,Z),mt(["w","ww","W","WW"],(function(t,e,n,i){e[i.substr(0,1)]=k(t)})),j("d",0,"do","day"),j("dd",0,0,(function(t){return this.localeData().weekdaysMin(this,t)})),j("ddd",0,0,(function(t){return this.localeData().weekdaysShort(this,t)})),j("dddd",0,0,(function(t){return this.localeData().weekdays(this,t)})),j("e",0,0,"weekday"),j("E",0,0,"isoWeekday"),L("day","d"),L("weekday","e"),L("isoWeekday","E"),Y("day",11),Y("weekday",11),Y("isoWeekday",11),dt("d",J),dt("e",J),dt("E",J),dt("dd",(function(t,e){return e.weekdaysMinRegex(t)})),dt("ddd",(function(t,e){return e.weekdaysShortRegex(t)})),dt("dddd",(function(t,e){return e.weekdaysRegex(t)})),mt(["dd","ddd","dddd"],(function(t,e,n,i){var a=n._locale.weekdaysParse(t,i,n._strict);null!=a?e.d=a:g(n).invalidWeekday=t})),mt(["d","e","E"],(function(t,e,n,i){e[i]=k(t)}));var Xt="Sunday_Monday_Tuesday_Wednesday_Thursday_Friday_Saturday".split("_"),Kt="Sun_Mon_Tue_Wed_Thu_Fri_Sat".split("_"),Jt="Su_Mo_Tu_We_Th_Fr_Sa".split("_");function Qt(t,e,n){var i,a,r,o=t.toLocaleLowerCase();if(!this._weekdaysParse)for(this._weekdaysParse=[],this._shortWeekdaysParse=[],this._minWeekdaysParse=[],i=0;i<7;++i)r=f([2e3,1]).day(i),this._minWeekdaysParse[i]=this.weekdaysMin(r,"").toLocaleLowerCase(),this._shortWeekdaysParse[i]=this.weekdaysShort(r,"").toLocaleLowerCase(),this._weekdaysParse[i]=this.weekdays(r,"").toLocaleLowerCase();return n?"dddd"===e?-1!==(a=Pt.call(this._weekdaysParse,o))?a:null:"ddd"===e?-1!==(a=Pt.call(this._shortWeekdaysParse,o))?a:null:-1!==(a=Pt.call(this._minWeekdaysParse,o))?a:null:"dddd"===e?-1!==(a=Pt.call(this._weekdaysParse,o))?a:-1!==(a=Pt.call(this._shortWeekdaysParse,o))?a:-1!==(a=Pt.call(this._minWeekdaysParse,o))?a:null:"ddd"===e?-1!==(a=Pt.call(this._shortWeekdaysParse,o))?a:-1!==(a=Pt.call(this._weekdaysParse,o))?a:-1!==(a=Pt.call(this._minWeekdaysParse,o))?a:null:-1!==(a=Pt.call(this._minWeekdaysParse,o))?a:-1!==(a=Pt.call(this._weekdaysParse,o))?a:-1!==(a=Pt.call(this._shortWeekdaysParse,o))?a:null}var te=lt,ee=lt,ne=lt;function ie(){function t(t,e){return e.length-t.length}var e,n,i,a,r,o=[],s=[],l=[],u=[];for(e=0;e<7;e++)n=f([2e3,1]).day(e),i=this.weekdaysMin(n,""),a=this.weekdaysShort(n,""),r=this.weekdays(n,""),o.push(i),s.push(a),l.push(r),u.push(i),u.push(a),u.push(r);for(o.sort(t),s.sort(t),l.sort(t),u.sort(t),e=0;e<7;e++)s[e]=ct(s[e]),l[e]=ct(l[e]),u[e]=ct(u[e]);this._weekdaysRegex=new RegExp("^("+u.join("|")+")","i"),this._weekdaysShortRegex=this._weekdaysRegex,this._weekdaysMinRegex=this._weekdaysRegex,this._weekdaysStrictRegex=new RegExp("^("+l.join("|")+")","i"),this._weekdaysShortStrictRegex=new RegExp("^("+s.join("|")+")","i"),this._weekdaysMinStrictRegex=new RegExp("^("+o.join("|")+")","i")}function ae(){return this.hours()%12||12}function re(t,e){j(t,0,0,(function(){return this.localeData().meridiem(this.hours(),this.minutes(),e)}))}function oe(t,e){return e._meridiemParse}j("H",["HH",2],0,"hour"),j("h",["hh",2],0,ae),j("k",["kk",2],0,(function(){return this.hours()||24})),j("hmm",0,0,(function(){return""+ae.apply(this)+z(this.minutes(),2)})),j("hmmss",0,0,(function(){return""+ae.apply(this)+z(this.minutes(),2)+z(this.seconds(),2)})),j("Hmm",0,0,(function(){return""+this.hours()+z(this.minutes(),2)})),j("Hmmss",0,0,(function(){return""+this.hours()+z(this.minutes(),2)+z(this.seconds(),2)})),re("a",!0),re("A",!1),L("hour","h"),Y("hour",13),dt("a",oe),dt("A",oe),dt("H",J),dt("h",J),dt("k",J),dt("HH",J,Z),dt("hh",J,Z),dt("kk",J,Z),dt("hmm",Q),dt("hmmss",tt),dt("Hmm",Q),dt("Hmmss",tt),gt(["H","HH"],xt),gt(["k","kk"],(function(t,e,n){var i=k(t);e[xt]=24===i?0:i})),gt(["a","A"],(function(t,e,n){n._isPm=n._locale.isPM(t),n._meridiem=t})),gt(["h","hh"],(function(t,e,n){e[xt]=k(t),g(n).bigHour=!0})),gt("hmm",(function(t,e,n){var i=t.length-2;e[xt]=k(t.substr(0,i)),e[_t]=k(t.substr(i)),g(n).bigHour=!0})),gt("hmmss",(function(t,e,n){var i=t.length-4,a=t.length-2;e[xt]=k(t.substr(0,i)),e[_t]=k(t.substr(i,2)),e[wt]=k(t.substr(a)),g(n).bigHour=!0})),gt("Hmm",(function(t,e,n){var i=t.length-2;e[xt]=k(t.substr(0,i)),e[_t]=k(t.substr(i))})),gt("Hmmss",(function(t,e,n){var i=t.length-4,a=t.length-2;e[xt]=k(t.substr(0,i)),e[_t]=k(t.substr(i,2)),e[wt]=k(t.substr(a))}));var se,le=Ot("Hours",!0),ue={calendar:{sameDay:"[Today at] LT",nextDay:"[Tomorrow at] LT",nextWeek:"dddd [at] LT",lastDay:"[Yesterday at] LT",lastWeek:"[Last] dddd [at] LT",sameElse:"L"},longDateFormat:{LTS:"h:mm:ss A",LT:"h:mm A",L:"MM/DD/YYYY",LL:"MMMM D, YYYY",LLL:"MMMM D, YYYY h:mm A",LLLL:"dddd, MMMM D, YYYY h:mm A"},invalidDate:"Invalid date",ordinal:"%d",dayOfMonthOrdinalParse:/\d{1,2}/,relativeTime:{future:"in %s",past:"%s ago",s:"a few seconds",ss:"%d seconds",m:"a minute",mm:"%d minutes",h:"an hour",hh:"%d hours",d:"a day",dd:"%d days",M:"a month",MM:"%d months",y:"a year",yy:"%d years"},months:Rt,monthsShort:Nt,week:{dow:0,doy:6},weekdays:Xt,weekdaysMin:Jt,weekdaysShort:Kt,meridiemParse:/[ap]\.?m?\.?/i},de={},he={};function ce(t){return t?t.toLowerCase().replace("_","-"):t}function fe(n){var i=null;if(!de[n]&&e&&e.exports)try{i=se._abbr,t(),ge(i)}catch(t){}return de[n]}function ge(t,e){var n;return t&&((n=s(e)?pe(t):me(t,e))?se=n:"undefined"!=typeof console&&console.warn&&console.warn("Locale "+t+" not found. Did you forget to load it?")),se._abbr}function me(t,e){if(null!==e){var n,i=ue;if(e.abbr=t,null!=de[t])T("defineLocaleOverride","use moment.updateLocale(localeName, config) to change an existing locale. moment.defineLocale(localeName, config) should only be used for creating a new locale See http://momentjs.com/guides/#/warnings/define-locale/ for more info."),i=de[t]._config;else if(null!=e.parentLocale)if(null!=de[e.parentLocale])i=de[e.parentLocale]._config;else{if(null==(n=fe(e.parentLocale)))return he[e.parentLocale]||(he[e.parentLocale]=[]),he[e.parentLocale].push({name:t,config:e}),null;i=n._config}return de[t]=new F(A(i,e)),he[t]&&he[t].forEach((function(t){me(t.name,t.config)})),ge(t),de[t]}return delete de[t],null}function pe(t){var e;if(t&&t._locale&&t._locale._abbr&&(t=t._locale._abbr),!t)return se;if(!r(t)){if(e=fe(t))return e;t=[t]}return function(t){for(var e,n,i,a,r=0;r0;){if(i=fe(a.slice(0,e).join("-")))return i;if(n&&n.length>=e&&M(a,n,!0)>=e-1)break;e--}r++}return se}(t)}function ve(t){var e,n=t._a;return n&&-2===g(t).overflow&&(e=n[bt]<0||n[bt]>11?bt:n[yt]<1||n[yt]>It(n[vt],n[bt])?yt:n[xt]<0||n[xt]>24||24===n[xt]&&(0!==n[_t]||0!==n[wt]||0!==n[kt])?xt:n[_t]<0||n[_t]>59?_t:n[wt]<0||n[wt]>59?wt:n[kt]<0||n[kt]>999?kt:-1,g(t)._overflowDayOfYear&&(eyt)&&(e=yt),g(t)._overflowWeeks&&-1===e&&(e=Mt),g(t)._overflowWeekday&&-1===e&&(e=St),g(t).overflow=e),t}function be(t,e,n){return null!=t?t:null!=e?e:n}function ye(t){var e,n,i,r,o,s=[];if(!t._d){for(i=function(t){var e=new Date(a.now());return t._useUTC?[e.getUTCFullYear(),e.getUTCMonth(),e.getUTCDate()]:[e.getFullYear(),e.getMonth(),e.getDate()]}(t),t._w&&null==t._a[yt]&&null==t._a[bt]&&function(t){var e,n,i,a,r,o,s,l;if(null!=(e=t._w).GG||null!=e.W||null!=e.E)r=1,o=4,n=be(e.GG,t._a[vt],qt(Le(),1,4).year),i=be(e.W,1),((a=be(e.E,1))<1||a>7)&&(l=!0);else{r=t._locale._week.dow,o=t._locale._week.doy;var u=qt(Le(),r,o);n=be(e.gg,t._a[vt],u.year),i=be(e.w,u.week),null!=e.d?((a=e.d)<0||a>6)&&(l=!0):null!=e.e?(a=e.e+r,(e.e<0||e.e>6)&&(l=!0)):a=r}i<1||i>Zt(n,r,o)?g(t)._overflowWeeks=!0:null!=l?g(t)._overflowWeekday=!0:(s=Gt(n,i,a,r,o),t._a[vt]=s.year,t._dayOfYear=s.dayOfYear)}(t),null!=t._dayOfYear&&(o=be(t._a[vt],i[vt]),(t._dayOfYear>Dt(o)||0===t._dayOfYear)&&(g(t)._overflowDayOfYear=!0),n=jt(o,0,t._dayOfYear),t._a[bt]=n.getUTCMonth(),t._a[yt]=n.getUTCDate()),e=0;e<3&&null==t._a[e];++e)t._a[e]=s[e]=i[e];for(;e<7;e++)t._a[e]=s[e]=null==t._a[e]?2===e?1:0:t._a[e];24===t._a[xt]&&0===t._a[_t]&&0===t._a[wt]&&0===t._a[kt]&&(t._nextDay=!0,t._a[xt]=0),t._d=(t._useUTC?jt:Bt).apply(null,s),r=t._useUTC?t._d.getUTCDay():t._d.getDay(),null!=t._tzm&&t._d.setUTCMinutes(t._d.getUTCMinutes()-t._tzm),t._nextDay&&(t._a[xt]=24),t._w&&void 0!==t._w.d&&t._w.d!==r&&(g(t).weekdayMismatch=!0)}}var xe=/^\s*((?:[+-]\d{6}|\d{4})-(?:\d\d-\d\d|W\d\d-\d|W\d\d|\d\d\d|\d\d))(?:(T| )(\d\d(?::\d\d(?::\d\d(?:[.,]\d+)?)?)?)([\+\-]\d\d(?::?\d\d)?|\s*Z)?)?$/,_e=/^\s*((?:[+-]\d{6}|\d{4})(?:\d\d\d\d|W\d\d\d|W\d\d|\d\d\d|\d\d))(?:(T| )(\d\d(?:\d\d(?:\d\d(?:[.,]\d+)?)?)?)([\+\-]\d\d(?::?\d\d)?|\s*Z)?)?$/,we=/Z|[+-]\d\d(?::?\d\d)?/,ke=[["YYYYYY-MM-DD",/[+-]\d{6}-\d\d-\d\d/],["YYYY-MM-DD",/\d{4}-\d\d-\d\d/],["GGGG-[W]WW-E",/\d{4}-W\d\d-\d/],["GGGG-[W]WW",/\d{4}-W\d\d/,!1],["YYYY-DDD",/\d{4}-\d{3}/],["YYYY-MM",/\d{4}-\d\d/,!1],["YYYYYYMMDD",/[+-]\d{10}/],["YYYYMMDD",/\d{8}/],["GGGG[W]WWE",/\d{4}W\d{3}/],["GGGG[W]WW",/\d{4}W\d{2}/,!1],["YYYYDDD",/\d{7}/]],Me=[["HH:mm:ss.SSSS",/\d\d:\d\d:\d\d\.\d+/],["HH:mm:ss,SSSS",/\d\d:\d\d:\d\d,\d+/],["HH:mm:ss",/\d\d:\d\d:\d\d/],["HH:mm",/\d\d:\d\d/],["HHmmss.SSSS",/\d\d\d\d\d\d\.\d+/],["HHmmss,SSSS",/\d\d\d\d\d\d,\d+/],["HHmmss",/\d\d\d\d\d\d/],["HHmm",/\d\d\d\d/],["HH",/\d\d/]],Se=/^\/?Date\((\-?\d+)/i;function De(t){var e,n,i,a,r,o,s=t._i,l=xe.exec(s)||_e.exec(s);if(l){for(g(t).iso=!0,e=0,n=ke.length;e0&&g(t).unusedInput.push(o),s=s.slice(s.indexOf(n)+n.length),u+=n.length),B[r]?(n?g(t).empty=!1:g(t).unusedTokens.push(r),pt(r,n,t)):t._strict&&!n&&g(t).unusedTokens.push(r);g(t).charsLeftOver=l-u,s.length>0&&g(t).unusedInput.push(s),t._a[xt]<=12&&!0===g(t).bigHour&&t._a[xt]>0&&(g(t).bigHour=void 0),g(t).parsedDateParts=t._a.slice(0),g(t).meridiem=t._meridiem,t._a[xt]=function(t,e,n){var i;return null==n?e:null!=t.meridiemHour?t.meridiemHour(e,n):null!=t.isPM?((i=t.isPM(n))&&e<12&&(e+=12),i||12!==e||(e=0),e):e}(t._locale,t._a[xt],t._meridiem),ye(t),ve(t)}else Oe(t);else De(t)}function Fe(t){var e=t._i,n=t._f;return t._locale=t._locale||pe(t._l),null===e||void 0===n&&""===e?p({nullInput:!0}):("string"==typeof e&&(t._i=e=t._locale.preparse(e)),_(e)?new x(ve(e)):(u(e)?t._d=e:r(n)?function(t){var e,n,i,a,r;if(0===t._f.length)return g(t).invalidFormat=!0,void(t._d=new Date(NaN));for(a=0;athis?this:t:p()}));function We(t,e){var n,i;if(1===e.length&&r(e[0])&&(e=e[0]),!e.length)return Le();for(n=e[0],i=1;i=0?new Date(t+400,e,n)-hn:new Date(t,e,n).valueOf()}function gn(t,e,n){return t<100&&t>=0?Date.UTC(t+400,e,n)-hn:Date.UTC(t,e,n)}function mn(t,e){j(0,[t,t.length],0,e)}function pn(t,e,n,i,a){var r;return null==t?qt(this,i,a).year:(e>(r=Zt(t,i,a))&&(e=r),vn.call(this,t,e,n,i,a))}function vn(t,e,n,i,a){var r=Gt(t,e,n,i,a),o=jt(r.year,0,r.dayOfYear);return this.year(o.getUTCFullYear()),this.month(o.getUTCMonth()),this.date(o.getUTCDate()),this}j(0,["gg",2],0,(function(){return this.weekYear()%100})),j(0,["GG",2],0,(function(){return this.isoWeekYear()%100})),mn("gggg","weekYear"),mn("ggggg","weekYear"),mn("GGGG","isoWeekYear"),mn("GGGGG","isoWeekYear"),L("weekYear","gg"),L("isoWeekYear","GG"),Y("weekYear",1),Y("isoWeekYear",1),dt("G",rt),dt("g",rt),dt("GG",J,Z),dt("gg",J,Z),dt("GGGG",nt,X),dt("gggg",nt,X),dt("GGGGG",it,K),dt("ggggg",it,K),mt(["gggg","ggggg","GGGG","GGGGG"],(function(t,e,n,i){e[i.substr(0,2)]=k(t)})),mt(["gg","GG"],(function(t,e,n,i){e[i]=a.parseTwoDigitYear(t)})),j("Q",0,"Qo","quarter"),L("quarter","Q"),Y("quarter",7),dt("Q",q),gt("Q",(function(t,e){e[bt]=3*(k(t)-1)})),j("D",["DD",2],"Do","date"),L("date","D"),Y("date",9),dt("D",J),dt("DD",J,Z),dt("Do",(function(t,e){return t?e._dayOfMonthOrdinalParse||e._ordinalParse:e._dayOfMonthOrdinalParseLenient})),gt(["D","DD"],yt),gt("Do",(function(t,e){e[yt]=k(t.match(J)[0])}));var bn=Ot("Date",!0);j("DDD",["DDDD",3],"DDDo","dayOfYear"),L("dayOfYear","DDD"),Y("dayOfYear",4),dt("DDD",et),dt("DDDD",$),gt(["DDD","DDDD"],(function(t,e,n){n._dayOfYear=k(t)})),j("m",["mm",2],0,"minute"),L("minute","m"),Y("minute",14),dt("m",J),dt("mm",J,Z),gt(["m","mm"],_t);var yn=Ot("Minutes",!1);j("s",["ss",2],0,"second"),L("second","s"),Y("second",15),dt("s",J),dt("ss",J,Z),gt(["s","ss"],wt);var xn,_n=Ot("Seconds",!1);for(j("S",0,0,(function(){return~~(this.millisecond()/100)})),j(0,["SS",2],0,(function(){return~~(this.millisecond()/10)})),j(0,["SSS",3],0,"millisecond"),j(0,["SSSS",4],0,(function(){return 10*this.millisecond()})),j(0,["SSSSS",5],0,(function(){return 100*this.millisecond()})),j(0,["SSSSSS",6],0,(function(){return 1e3*this.millisecond()})),j(0,["SSSSSSS",7],0,(function(){return 1e4*this.millisecond()})),j(0,["SSSSSSSS",8],0,(function(){return 1e5*this.millisecond()})),j(0,["SSSSSSSSS",9],0,(function(){return 1e6*this.millisecond()})),L("millisecond","ms"),Y("millisecond",16),dt("S",et,q),dt("SS",et,Z),dt("SSS",et,$),xn="SSSS";xn.length<=9;xn+="S")dt(xn,at);function wn(t,e){e[kt]=k(1e3*("0."+t))}for(xn="S";xn.length<=9;xn+="S")gt(xn,wn);var kn=Ot("Milliseconds",!1);j("z",0,0,"zoneAbbr"),j("zz",0,0,"zoneName");var Mn=x.prototype;function Sn(t){return t}Mn.add=en,Mn.calendar=function(t,e){var n=t||Le(),i=Ue(n,this).startOf("day"),r=a.calendarFormat(this,i)||"sameElse",o=e&&(O(e[r])?e[r].call(this,n):e[r]);return this.format(o||this.localeData().calendar(r,this,Le(n)))},Mn.clone=function(){return new x(this)},Mn.diff=function(t,e,n){var i,a,r;if(!this.isValid())return NaN;if(!(i=Ue(t,this)).isValid())return NaN;switch(a=6e4*(i.utcOffset()-this.utcOffset()),e=R(e)){case"year":r=an(this,i)/12;break;case"month":r=an(this,i);break;case"quarter":r=an(this,i)/3;break;case"second":r=(this-i)/1e3;break;case"minute":r=(this-i)/6e4;break;case"hour":r=(this-i)/36e5;break;case"day":r=(this-i-a)/864e5;break;case"week":r=(this-i-a)/6048e5;break;default:r=this-i}return n?r:w(r)},Mn.endOf=function(t){var e;if(void 0===(t=R(t))||"millisecond"===t||!this.isValid())return this;var n=this._isUTC?gn:fn;switch(t){case"year":e=n(this.year()+1,0,1)-1;break;case"quarter":e=n(this.year(),this.month()-this.month()%3+3,1)-1;break;case"month":e=n(this.year(),this.month()+1,1)-1;break;case"week":e=n(this.year(),this.month(),this.date()-this.weekday()+7)-1;break;case"isoWeek":e=n(this.year(),this.month(),this.date()-(this.isoWeekday()-1)+7)-1;break;case"day":case"date":e=n(this.year(),this.month(),this.date()+1)-1;break;case"hour":e=this._d.valueOf(),e+=dn-cn(e+(this._isUTC?0:this.utcOffset()*un),dn)-1;break;case"minute":e=this._d.valueOf(),e+=un-cn(e,un)-1;break;case"second":e=this._d.valueOf(),e+=ln-cn(e,ln)-1}return this._d.setTime(e),a.updateOffset(this,!0),this},Mn.format=function(t){t||(t=this.isUtc()?a.defaultFormatUtc:a.defaultFormat);var e=U(this,t);return this.localeData().postformat(e)},Mn.from=function(t,e){return this.isValid()&&(_(t)&&t.isValid()||Le(t).isValid())?Xe({to:this,from:t}).locale(this.locale()).humanize(!e):this.localeData().invalidDate()},Mn.fromNow=function(t){return this.from(Le(),t)},Mn.to=function(t,e){return this.isValid()&&(_(t)&&t.isValid()||Le(t).isValid())?Xe({from:this,to:t}).locale(this.locale()).humanize(!e):this.localeData().invalidDate()},Mn.toNow=function(t){return this.to(Le(),t)},Mn.get=function(t){return O(this[t=R(t)])?this[t]():this},Mn.invalidAt=function(){return g(this).overflow},Mn.isAfter=function(t,e){var n=_(t)?t:Le(t);return!(!this.isValid()||!n.isValid())&&("millisecond"===(e=R(e)||"millisecond")?this.valueOf()>n.valueOf():n.valueOf()9999?U(n,e?"YYYYYY-MM-DD[T]HH:mm:ss.SSS[Z]":"YYYYYY-MM-DD[T]HH:mm:ss.SSSZ"):O(Date.prototype.toISOString)?e?this.toDate().toISOString():new Date(this.valueOf()+60*this.utcOffset()*1e3).toISOString().replace("Z",U(n,"Z")):U(n,e?"YYYY-MM-DD[T]HH:mm:ss.SSS[Z]":"YYYY-MM-DD[T]HH:mm:ss.SSSZ")},Mn.inspect=function(){if(!this.isValid())return"moment.invalid(/* "+this._i+" */)";var t="moment",e="";this.isLocal()||(t=0===this.utcOffset()?"moment.utc":"moment.parseZone",e="Z");var n="["+t+'("]',i=0<=this.year()&&this.year()<=9999?"YYYY":"YYYYYY",a=e+'[")]';return this.format(n+i+"-MM-DD[T]HH:mm:ss.SSS"+a)},Mn.toJSON=function(){return this.isValid()?this.toISOString():null},Mn.toString=function(){return this.clone().locale("en").format("ddd MMM DD YYYY HH:mm:ss [GMT]ZZ")},Mn.unix=function(){return Math.floor(this.valueOf()/1e3)},Mn.valueOf=function(){return this._d.valueOf()-6e4*(this._offset||0)},Mn.creationData=function(){return{input:this._i,format:this._f,locale:this._locale,isUTC:this._isUTC,strict:this._strict}},Mn.year=Tt,Mn.isLeapYear=function(){return Ct(this.year())},Mn.weekYear=function(t){return pn.call(this,t,this.week(),this.weekday(),this.localeData()._week.dow,this.localeData()._week.doy)},Mn.isoWeekYear=function(t){return pn.call(this,t,this.isoWeek(),this.isoWeekday(),1,4)},Mn.quarter=Mn.quarters=function(t){return null==t?Math.ceil((this.month()+1)/3):this.month(3*(t-1)+this.month()%3)},Mn.month=zt,Mn.daysInMonth=function(){return It(this.year(),this.month())},Mn.week=Mn.weeks=function(t){var e=this.localeData().week(this);return null==t?e:this.add(7*(t-e),"d")},Mn.isoWeek=Mn.isoWeeks=function(t){var e=qt(this,1,4).week;return null==t?e:this.add(7*(t-e),"d")},Mn.weeksInYear=function(){var t=this.localeData()._week;return Zt(this.year(),t.dow,t.doy)},Mn.isoWeeksInYear=function(){return Zt(this.year(),1,4)},Mn.date=bn,Mn.day=Mn.days=function(t){if(!this.isValid())return null!=t?this:NaN;var e=this._isUTC?this._d.getUTCDay():this._d.getDay();return null!=t?(t=function(t,e){return"string"!=typeof t?t:isNaN(t)?"number"==typeof(t=e.weekdaysParse(t))?t:null:parseInt(t,10)}(t,this.localeData()),this.add(t-e,"d")):e},Mn.weekday=function(t){if(!this.isValid())return null!=t?this:NaN;var e=(this.day()+7-this.localeData()._week.dow)%7;return null==t?e:this.add(t-e,"d")},Mn.isoWeekday=function(t){if(!this.isValid())return null!=t?this:NaN;if(null!=t){var e=function(t,e){return"string"==typeof t?e.weekdaysParse(t)%7||7:isNaN(t)?null:t}(t,this.localeData());return this.day(this.day()%7?e:e-7)}return this.day()||7},Mn.dayOfYear=function(t){var e=Math.round((this.clone().startOf("day")-this.clone().startOf("year"))/864e5)+1;return null==t?e:this.add(t-e,"d")},Mn.hour=Mn.hours=le,Mn.minute=Mn.minutes=yn,Mn.second=Mn.seconds=_n,Mn.millisecond=Mn.milliseconds=kn,Mn.utcOffset=function(t,e,n){var i,r=this._offset||0;if(!this.isValid())return null!=t?this:NaN;if(null!=t){if("string"==typeof t){if(null===(t=je(st,t)))return this}else Math.abs(t)<16&&!n&&(t*=60);return!this._isUTC&&e&&(i=Ge(this)),this._offset=t,this._isUTC=!0,null!=i&&this.add(i,"m"),r!==t&&(!e||this._changeInProgress?tn(this,Xe(t-r,"m"),1,!1):this._changeInProgress||(this._changeInProgress=!0,a.updateOffset(this,!0),this._changeInProgress=null)),this}return this._isUTC?r:Ge(this)},Mn.utc=function(t){return this.utcOffset(0,t)},Mn.local=function(t){return this._isUTC&&(this.utcOffset(0,t),this._isUTC=!1,t&&this.subtract(Ge(this),"m")),this},Mn.parseZone=function(){if(null!=this._tzm)this.utcOffset(this._tzm,!1,!0);else if("string"==typeof this._i){var t=je(ot,this._i);null!=t?this.utcOffset(t):this.utcOffset(0,!0)}return this},Mn.hasAlignedHourOffset=function(t){return!!this.isValid()&&(t=t?Le(t).utcOffset():0,(this.utcOffset()-t)%60==0)},Mn.isDST=function(){return this.utcOffset()>this.clone().month(0).utcOffset()||this.utcOffset()>this.clone().month(5).utcOffset()},Mn.isLocal=function(){return!!this.isValid()&&!this._isUTC},Mn.isUtcOffset=function(){return!!this.isValid()&&this._isUTC},Mn.isUtc=qe,Mn.isUTC=qe,Mn.zoneAbbr=function(){return this._isUTC?"UTC":""},Mn.zoneName=function(){return this._isUTC?"Coordinated Universal Time":""},Mn.dates=D("dates accessor is deprecated. Use date instead.",bn),Mn.months=D("months accessor is deprecated. Use month instead",zt),Mn.years=D("years accessor is deprecated. Use year instead",Tt),Mn.zone=D("moment().zone is deprecated, use moment().utcOffset instead. http://momentjs.com/guides/#/warnings/zone/",(function(t,e){return null!=t?("string"!=typeof t&&(t=-t),this.utcOffset(t,e),this):-this.utcOffset()})),Mn.isDSTShifted=D("isDSTShifted is deprecated. See http://momentjs.com/guides/#/warnings/dst-shifted/ for more information",(function(){if(!s(this._isDSTShifted))return this._isDSTShifted;var t={};if(b(t,this),(t=Fe(t))._a){var e=t._isUTC?f(t._a):Le(t._a);this._isDSTShifted=this.isValid()&&M(t._a,e.toArray())>0}else this._isDSTShifted=!1;return this._isDSTShifted}));var Dn=F.prototype;function Cn(t,e,n,i){var a=pe(),r=f().set(i,e);return a[n](r,t)}function Pn(t,e,n){if(l(t)&&(e=t,t=void 0),t=t||"",null!=e)return Cn(t,e,n,"month");var i,a=[];for(i=0;i<12;i++)a[i]=Cn(t,i,n,"month");return a}function Tn(t,e,n,i){"boolean"==typeof t?(l(e)&&(n=e,e=void 0),e=e||""):(n=e=t,t=!1,l(e)&&(n=e,e=void 0),e=e||"");var a,r=pe(),o=t?r._week.dow:0;if(null!=n)return Cn(e,(n+o)%7,i,"day");var s=[];for(a=0;a<7;a++)s[a]=Cn(e,(a+o)%7,i,"day");return s}Dn.calendar=function(t,e,n){var i=this._calendar[t]||this._calendar.sameElse;return O(i)?i.call(e,n):i},Dn.longDateFormat=function(t){var e=this._longDateFormat[t],n=this._longDateFormat[t.toUpperCase()];return e||!n?e:(this._longDateFormat[t]=n.replace(/MMMM|MM|DD|dddd/g,(function(t){return t.slice(1)})),this._longDateFormat[t])},Dn.invalidDate=function(){return this._invalidDate},Dn.ordinal=function(t){return this._ordinal.replace("%d",t)},Dn.preparse=Sn,Dn.postformat=Sn,Dn.relativeTime=function(t,e,n,i){var a=this._relativeTime[n];return O(a)?a(t,e,n,i):a.replace(/%d/i,t)},Dn.pastFuture=function(t,e){var n=this._relativeTime[t>0?"future":"past"];return O(n)?n(e):n.replace(/%s/i,e)},Dn.set=function(t){var e,n;for(n in t)O(e=t[n])?this[n]=e:this["_"+n]=e;this._config=t,this._dayOfMonthOrdinalParseLenient=new RegExp((this._dayOfMonthOrdinalParse.source||this._ordinalParse.source)+"|"+/\d{1,2}/.source)},Dn.months=function(t,e){return t?r(this._months)?this._months[t.month()]:this._months[(this._months.isFormat||Lt).test(e)?"format":"standalone"][t.month()]:r(this._months)?this._months:this._months.standalone},Dn.monthsShort=function(t,e){return t?r(this._monthsShort)?this._monthsShort[t.month()]:this._monthsShort[Lt.test(e)?"format":"standalone"][t.month()]:r(this._monthsShort)?this._monthsShort:this._monthsShort.standalone},Dn.monthsParse=function(t,e,n){var i,a,r;if(this._monthsParseExact)return Wt.call(this,t,e,n);for(this._monthsParse||(this._monthsParse=[],this._longMonthsParse=[],this._shortMonthsParse=[]),i=0;i<12;i++){if(a=f([2e3,i]),n&&!this._longMonthsParse[i]&&(this._longMonthsParse[i]=new RegExp("^"+this.months(a,"").replace(".","")+"$","i"),this._shortMonthsParse[i]=new RegExp("^"+this.monthsShort(a,"").replace(".","")+"$","i")),n||this._monthsParse[i]||(r="^"+this.months(a,"")+"|^"+this.monthsShort(a,""),this._monthsParse[i]=new RegExp(r.replace(".",""),"i")),n&&"MMMM"===e&&this._longMonthsParse[i].test(t))return i;if(n&&"MMM"===e&&this._shortMonthsParse[i].test(t))return i;if(!n&&this._monthsParse[i].test(t))return i}},Dn.monthsRegex=function(t){return this._monthsParseExact?(h(this,"_monthsRegex")||Ht.call(this),t?this._monthsStrictRegex:this._monthsRegex):(h(this,"_monthsRegex")||(this._monthsRegex=Vt),this._monthsStrictRegex&&t?this._monthsStrictRegex:this._monthsRegex)},Dn.monthsShortRegex=function(t){return this._monthsParseExact?(h(this,"_monthsRegex")||Ht.call(this),t?this._monthsShortStrictRegex:this._monthsShortRegex):(h(this,"_monthsShortRegex")||(this._monthsShortRegex=Et),this._monthsShortStrictRegex&&t?this._monthsShortStrictRegex:this._monthsShortRegex)},Dn.week=function(t){return qt(t,this._week.dow,this._week.doy).week},Dn.firstDayOfYear=function(){return this._week.doy},Dn.firstDayOfWeek=function(){return this._week.dow},Dn.weekdays=function(t,e){var n=r(this._weekdays)?this._weekdays:this._weekdays[t&&!0!==t&&this._weekdays.isFormat.test(e)?"format":"standalone"];return!0===t?$t(n,this._week.dow):t?n[t.day()]:n},Dn.weekdaysMin=function(t){return!0===t?$t(this._weekdaysMin,this._week.dow):t?this._weekdaysMin[t.day()]:this._weekdaysMin},Dn.weekdaysShort=function(t){return!0===t?$t(this._weekdaysShort,this._week.dow):t?this._weekdaysShort[t.day()]:this._weekdaysShort},Dn.weekdaysParse=function(t,e,n){var i,a,r;if(this._weekdaysParseExact)return Qt.call(this,t,e,n);for(this._weekdaysParse||(this._weekdaysParse=[],this._minWeekdaysParse=[],this._shortWeekdaysParse=[],this._fullWeekdaysParse=[]),i=0;i<7;i++){if(a=f([2e3,1]).day(i),n&&!this._fullWeekdaysParse[i]&&(this._fullWeekdaysParse[i]=new RegExp("^"+this.weekdays(a,"").replace(".","\\.?")+"$","i"),this._shortWeekdaysParse[i]=new RegExp("^"+this.weekdaysShort(a,"").replace(".","\\.?")+"$","i"),this._minWeekdaysParse[i]=new RegExp("^"+this.weekdaysMin(a,"").replace(".","\\.?")+"$","i")),this._weekdaysParse[i]||(r="^"+this.weekdays(a,"")+"|^"+this.weekdaysShort(a,"")+"|^"+this.weekdaysMin(a,""),this._weekdaysParse[i]=new RegExp(r.replace(".",""),"i")),n&&"dddd"===e&&this._fullWeekdaysParse[i].test(t))return i;if(n&&"ddd"===e&&this._shortWeekdaysParse[i].test(t))return i;if(n&&"dd"===e&&this._minWeekdaysParse[i].test(t))return i;if(!n&&this._weekdaysParse[i].test(t))return i}},Dn.weekdaysRegex=function(t){return this._weekdaysParseExact?(h(this,"_weekdaysRegex")||ie.call(this),t?this._weekdaysStrictRegex:this._weekdaysRegex):(h(this,"_weekdaysRegex")||(this._weekdaysRegex=te),this._weekdaysStrictRegex&&t?this._weekdaysStrictRegex:this._weekdaysRegex)},Dn.weekdaysShortRegex=function(t){return this._weekdaysParseExact?(h(this,"_weekdaysRegex")||ie.call(this),t?this._weekdaysShortStrictRegex:this._weekdaysShortRegex):(h(this,"_weekdaysShortRegex")||(this._weekdaysShortRegex=ee),this._weekdaysShortStrictRegex&&t?this._weekdaysShortStrictRegex:this._weekdaysShortRegex)},Dn.weekdaysMinRegex=function(t){return this._weekdaysParseExact?(h(this,"_weekdaysRegex")||ie.call(this),t?this._weekdaysMinStrictRegex:this._weekdaysMinRegex):(h(this,"_weekdaysMinRegex")||(this._weekdaysMinRegex=ne),this._weekdaysMinStrictRegex&&t?this._weekdaysMinStrictRegex:this._weekdaysMinRegex)},Dn.isPM=function(t){return"p"===(t+"").toLowerCase().charAt(0)},Dn.meridiem=function(t,e,n){return t>11?n?"pm":"PM":n?"am":"AM"},ge("en",{dayOfMonthOrdinalParse:/\d{1,2}(th|st|nd|rd)/,ordinal:function(t){var e=t%10;return t+(1===k(t%100/10)?"th":1===e?"st":2===e?"nd":3===e?"rd":"th")}}),a.lang=D("moment.lang is deprecated. Use moment.locale instead.",ge),a.langData=D("moment.langData is deprecated. Use moment.localeData instead.",pe);var On=Math.abs;function An(t,e,n,i){var a=Xe(e,n);return t._milliseconds+=i*a._milliseconds,t._days+=i*a._days,t._months+=i*a._months,t._bubble()}function Fn(t){return t<0?Math.floor(t):Math.ceil(t)}function In(t){return 4800*t/146097}function Ln(t){return 146097*t/4800}function Rn(t){return function(){return this.as(t)}}var Nn=Rn("ms"),Wn=Rn("s"),Yn=Rn("m"),zn=Rn("h"),En=Rn("d"),Vn=Rn("w"),Hn=Rn("M"),Bn=Rn("Q"),jn=Rn("y");function Un(t){return function(){return this.isValid()?this._data[t]:NaN}}var Gn=Un("milliseconds"),qn=Un("seconds"),Zn=Un("minutes"),$n=Un("hours"),Xn=Un("days"),Kn=Un("months"),Jn=Un("years"),Qn=Math.round,ti={ss:44,s:45,m:45,h:22,d:26,M:11};function ei(t,e,n,i,a){return a.relativeTime(e||1,!!n,t,i)}var ni=Math.abs;function ii(t){return(t>0)-(t<0)||+t}function ai(){if(!this.isValid())return this.localeData().invalidDate();var t,e,n=ni(this._milliseconds)/1e3,i=ni(this._days),a=ni(this._months);t=w(n/60),e=w(t/60),n%=60,t%=60;var r=w(a/12),o=a%=12,s=i,l=e,u=t,d=n?n.toFixed(3).replace(/\.?0+$/,""):"",h=this.asSeconds();if(!h)return"P0D";var c=h<0?"-":"",f=ii(this._months)!==ii(h)?"-":"",g=ii(this._days)!==ii(h)?"-":"",m=ii(this._milliseconds)!==ii(h)?"-":"";return c+"P"+(r?f+r+"Y":"")+(o?f+o+"M":"")+(s?g+s+"D":"")+(l||u||d?"T":"")+(l?m+l+"H":"")+(u?m+u+"M":"")+(d?m+d+"S":"")}var ri=ze.prototype;return ri.isValid=function(){return this._isValid},ri.abs=function(){var t=this._data;return this._milliseconds=On(this._milliseconds),this._days=On(this._days),this._months=On(this._months),t.milliseconds=On(t.milliseconds),t.seconds=On(t.seconds),t.minutes=On(t.minutes),t.hours=On(t.hours),t.months=On(t.months),t.years=On(t.years),this},ri.add=function(t,e){return An(this,t,e,1)},ri.subtract=function(t,e){return An(this,t,e,-1)},ri.as=function(t){if(!this.isValid())return NaN;var e,n,i=this._milliseconds;if("month"===(t=R(t))||"quarter"===t||"year"===t)switch(e=this._days+i/864e5,n=this._months+In(e),t){case"month":return n;case"quarter":return n/3;case"year":return n/12}else switch(e=this._days+Math.round(Ln(this._months)),t){case"week":return e/7+i/6048e5;case"day":return e+i/864e5;case"hour":return 24*e+i/36e5;case"minute":return 1440*e+i/6e4;case"second":return 86400*e+i/1e3;case"millisecond":return Math.floor(864e5*e)+i;default:throw new Error("Unknown unit "+t)}},ri.asMilliseconds=Nn,ri.asSeconds=Wn,ri.asMinutes=Yn,ri.asHours=zn,ri.asDays=En,ri.asWeeks=Vn,ri.asMonths=Hn,ri.asQuarters=Bn,ri.asYears=jn,ri.valueOf=function(){return this.isValid()?this._milliseconds+864e5*this._days+this._months%12*2592e6+31536e6*k(this._months/12):NaN},ri._bubble=function(){var t,e,n,i,a,r=this._milliseconds,o=this._days,s=this._months,l=this._data;return r>=0&&o>=0&&s>=0||r<=0&&o<=0&&s<=0||(r+=864e5*Fn(Ln(s)+o),o=0,s=0),l.milliseconds=r%1e3,t=w(r/1e3),l.seconds=t%60,e=w(t/60),l.minutes=e%60,n=w(e/60),l.hours=n%24,o+=w(n/24),a=w(In(o)),s+=a,o-=Fn(Ln(a)),i=w(s/12),s%=12,l.days=o,l.months=s,l.years=i,this},ri.clone=function(){return Xe(this)},ri.get=function(t){return t=R(t),this.isValid()?this[t+"s"]():NaN},ri.milliseconds=Gn,ri.seconds=qn,ri.minutes=Zn,ri.hours=$n,ri.days=Xn,ri.weeks=function(){return w(this.days()/7)},ri.months=Kn,ri.years=Jn,ri.humanize=function(t){if(!this.isValid())return this.localeData().invalidDate();var e=this.localeData(),n=function(t,e,n){var i=Xe(t).abs(),a=Qn(i.as("s")),r=Qn(i.as("m")),o=Qn(i.as("h")),s=Qn(i.as("d")),l=Qn(i.as("M")),u=Qn(i.as("y")),d=a<=ti.ss&&["s",a]||a0,d[4]=n,ei.apply(null,d)}(this,!t,e);return t&&(n=e.pastFuture(+this,n)),e.postformat(n)},ri.toISOString=ai,ri.toString=ai,ri.toJSON=ai,ri.locale=rn,ri.localeData=sn,ri.toIsoString=D("toIsoString() is deprecated. Please use toISOString() instead (notice the capitals)",ai),ri.lang=on,j("X",0,0,"unix"),j("x",0,0,"valueOf"),dt("x",rt),dt("X",/[+-]?\d+(\.\d{1,3})?/),gt("X",(function(t,e,n){n._d=new Date(1e3*parseFloat(t,10))})),gt("x",(function(t,e,n){n._d=new Date(k(t))})),a.version="2.24.0",n=Le,a.fn=Mn,a.min=function(){return We("isBefore",[].slice.call(arguments,0))},a.max=function(){return We("isAfter",[].slice.call(arguments,0))},a.now=function(){return Date.now?Date.now():+new Date},a.utc=f,a.unix=function(t){return Le(1e3*t)},a.months=function(t,e){return Pn(t,e,"months")},a.isDate=u,a.locale=ge,a.invalid=p,a.duration=Xe,a.isMoment=_,a.weekdays=function(t,e,n){return Tn(t,e,n,"weekdays")},a.parseZone=function(){return Le.apply(null,arguments).parseZone()},a.localeData=pe,a.isDuration=Ee,a.monthsShort=function(t,e){return Pn(t,e,"monthsShort")},a.weekdaysMin=function(t,e,n){return Tn(t,e,n,"weekdaysMin")},a.defineLocale=me,a.updateLocale=function(t,e){if(null!=e){var n,i,a=ue;null!=(i=fe(t))&&(a=i._config),e=A(a,e),(n=new F(e)).parentLocale=de[t],de[t]=n,ge(t)}else null!=de[t]&&(null!=de[t].parentLocale?de[t]=de[t].parentLocale:null!=de[t]&&delete de[t]);return de[t]},a.locales=function(){return C(de)},a.weekdaysShort=function(t,e,n){return Tn(t,e,n,"weekdaysShort")},a.normalizeUnits=R,a.relativeTimeRounding=function(t){return void 0===t?Qn:"function"==typeof t&&(Qn=t,!0)},a.relativeTimeThreshold=function(t,e){return void 0!==ti[t]&&(void 0===e?ti[t]:(ti[t]=e,"s"===t&&(ti.ss=e-1),!0))},a.calendarFormat=function(t,e){var n=t.diff(e,"days",!0);return n<-6?"sameElse":n<-1?"lastWeek":n<0?"lastDay":n<1?"sameDay":n<2?"nextDay":n<7?"nextWeek":"sameElse"},a.prototype=Mn,a.HTML5_FMT={DATETIME_LOCAL:"YYYY-MM-DDTHH:mm",DATETIME_LOCAL_SECONDS:"YYYY-MM-DDTHH:mm:ss",DATETIME_LOCAL_MS:"YYYY-MM-DDTHH:mm:ss.SSS",DATE:"YYYY-MM-DD",TIME:"HH:mm",TIME_SECONDS:"HH:mm:ss",TIME_MS:"HH:mm:ss.SSS",WEEK:"GGGG-[W]WW",MONTH:"YYYY-MM"},a}()})),mi={datetime:"MMM D, YYYY, h:mm:ss a",millisecond:"h:mm:ss.SSS a",second:"h:mm:ss a",minute:"h:mm a",hour:"hA",day:"MMM D",week:"ll",month:"MMM YYYY",quarter:"[Q]Q - YYYY",year:"YYYY"};rn._date.override("function"==typeof gi?{_id:"moment",formats:function(){return mi},parse:function(t,e){return"string"==typeof t&&"string"==typeof e?t=gi(t,e):t instanceof gi||(t=gi(t)),t.isValid()?t.valueOf():null},format:function(t,e){return gi(t).format(e)},add:function(t,e,n){return gi(t).add(e,n).valueOf()},diff:function(t,e,n){return gi(t).diff(gi(e),n)},startOf:function(t,e,n){return t=gi(t),"isoWeek"===e?t.isoWeekday(n).valueOf():t.startOf(e).valueOf()},endOf:function(t,e){return gi(t).endOf(e).valueOf()},_create:function(t){return gi(t)}}:{}),W._set("global",{plugins:{filler:{propagate:!0}}});var pi={dataset:function(t){var e=t.fill,n=t.chart,i=n.getDatasetMeta(e),a=i&&n.isDatasetVisible(e)&&i.dataset._children||[],r=a.length||0;return r?function(t,e){return e=n)&&i;switch(r){case"bottom":return"start";case"top":return"end";case"zero":return"origin";case"origin":case"start":case"end":return r;default:return!1}}function bi(t){return(t.el._scale||{}).getPointPositionForValue?function(t){var e,n,i,a,r,o=t.el._scale,s=o.options,l=o.chart.data.labels.length,u=t.fill,d=[];if(!l)return null;for(e=s.ticks.reverse?o.max:o.min,n=s.ticks.reverse?o.min:o.max,i=o.getPointPositionForValue(0,e),a=0;a0;--r)H.canvas.lineTo(t,n[r],n[r-1],!0);else for(o=n[0].cx,s=n[0].cy,l=Math.sqrt(Math.pow(n[0].x-o,2)+Math.pow(n[0].y-s,2)),r=a-1;r>0;--r)t.arc(o,s,l,n[r].angle,n[r-1].angle,!0)}}function ki(t,e,n,i,a,r){var o,s,l,u,d,h,c,f,g=e.length,m=i.spanGaps,p=[],v=[],b=0,y=0;for(t.beginPath(),o=0,s=g;o=0;--n)(e=l[n].$filler)&&e.visible&&(a=(i=e.el)._view,r=i._children||[],o=e.mapper,s=a.backgroundColor||W.global.defaultColor,o&&s&&r.length&&(H.canvas.clipArea(u,t.chartArea),ki(u,r,o,a,s,i._loop),H.canvas.unclipArea(u)))}},Si=H.rtl.getRtlAdapter,Di=H.noop,Ci=H.valueOrDefault;function Pi(t,e){return t.usePointStyle&&t.boxWidth>e?e:t.boxWidth}W._set("global",{legend:{display:!0,position:"top",align:"center",fullWidth:!0,reverse:!1,weight:1e3,onClick:function(t,e){var n=e.datasetIndex,i=this.chart,a=i.getDatasetMeta(n);a.hidden=null===a.hidden?!i.data.datasets[n].hidden:null,i.update()},onHover:null,onLeave:null,labels:{boxWidth:40,padding:10,generateLabels:function(t){var e=t.data.datasets,n=t.options.legend||{},i=n.labels&&n.labels.usePointStyle;return t._getSortedDatasetMetas().map((function(n){var a=n.controller.getStyle(i?0:void 0);return{text:e[n.index].label,fillStyle:a.backgroundColor,hidden:!t.isDatasetVisible(n.index),lineCap:a.borderCapStyle,lineDash:a.borderDash,lineDashOffset:a.borderDashOffset,lineJoin:a.borderJoinStyle,lineWidth:a.borderWidth,strokeStyle:a.borderColor,pointStyle:a.pointStyle,rotation:a.rotation,datasetIndex:n.index}}),this)}}},legendCallback:function(t){var e,n,i,a=document.createElement("ul"),r=t.data.datasets;for(a.setAttribute("class",t.id+"-legend"),e=0,n=r.length;el.width)&&(h+=o+n.padding,d[d.length-(e>0?0:1)]=0),s[e]={left:0,top:0,width:i,height:o},d[d.length-1]+=i+n.padding})),l.height+=h}else{var c=n.padding,f=t.columnWidths=[],g=t.columnHeights=[],m=n.padding,p=0,v=0;H.each(t.legendItems,(function(t,e){var i=Pi(n,o)+o/2+a.measureText(t.text).width;e>0&&v+o+2*c>l.height&&(m+=p+n.padding,f.push(p),g.push(v),p=0,v=0),p=Math.max(p,i),v+=o+c,s[e]={left:0,top:0,width:i,height:o}})),m+=p,f.push(p),g.push(v),l.width+=m}t.width=l.width,t.height=l.height}else t.width=l.width=t.height=l.height=0},afterFit:Di,isHorizontal:function(){return"top"===this.options.position||"bottom"===this.options.position},draw:function(){var t=this,e=t.options,n=e.labels,i=W.global,a=i.defaultColor,r=i.elements.line,o=t.height,s=t.columnHeights,l=t.width,u=t.lineWidths;if(e.display){var d,h=Si(e.rtl,t.left,t.minSize.width),c=t.ctx,f=Ci(n.fontColor,i.defaultFontColor),g=H.options._parseFont(n),m=g.size;c.textAlign=h.textAlign("left"),c.textBaseline="middle",c.lineWidth=.5,c.strokeStyle=f,c.fillStyle=f,c.font=g.string;var p=Pi(n,m),v=t.legendHitBoxes,b=function(t,i){switch(e.align){case"start":return n.padding;case"end":return t-i;default:return(t-i+n.padding)/2}},y=t.isHorizontal();d=y?{x:t.left+b(l,u[0]),y:t.top+n.padding,line:0}:{x:t.left+n.padding,y:t.top+b(o,s[0]),line:0},H.rtl.overrideTextDirection(t.ctx,e.textDirection);var x=m+n.padding;H.each(t.legendItems,(function(e,i){var f=c.measureText(e.text).width,g=p+m/2+f,_=d.x,w=d.y;h.setWidth(t.minSize.width),y?i>0&&_+g+n.padding>t.left+t.minSize.width&&(w=d.y+=x,d.line++,_=d.x=t.left+b(l,u[d.line])):i>0&&w+x>t.top+t.minSize.height&&(_=d.x=_+t.columnWidths[d.line]+n.padding,d.line++,w=d.y=t.top+b(o,s[d.line]));var k=h.x(_);!function(t,e,i){if(!(isNaN(p)||p<=0)){c.save();var o=Ci(i.lineWidth,r.borderWidth);if(c.fillStyle=Ci(i.fillStyle,a),c.lineCap=Ci(i.lineCap,r.borderCapStyle),c.lineDashOffset=Ci(i.lineDashOffset,r.borderDashOffset),c.lineJoin=Ci(i.lineJoin,r.borderJoinStyle),c.lineWidth=o,c.strokeStyle=Ci(i.strokeStyle,a),c.setLineDash&&c.setLineDash(Ci(i.lineDash,r.borderDash)),n&&n.usePointStyle){var s=p*Math.SQRT2/2,l=h.xPlus(t,p/2),u=e+m/2;H.canvas.drawPoint(c,i.pointStyle,s,l,u,i.rotation)}else c.fillRect(h.leftForLtr(t,p),e,p,m),0!==o&&c.strokeRect(h.leftForLtr(t,p),e,p,m);c.restore()}}(k,w,e),v[i].left=h.leftForLtr(k,v[i].width),v[i].top=w,function(t,e,n,i){var a=m/2,r=h.xPlus(t,p+a),o=e+a;c.fillText(n.text,r,o),n.hidden&&(c.beginPath(),c.lineWidth=2,c.moveTo(r,o),c.lineTo(h.xPlus(r,i),o),c.stroke())}(k,w,e,f),y?d.x+=g+n.padding:d.y+=x})),H.rtl.restoreTextDirection(t.ctx,e.textDirection)}},_getLegendItemAt:function(t,e){var n,i,a,r=this;if(t>=r.left&&t<=r.right&&e>=r.top&&e<=r.bottom)for(a=r.legendHitBoxes,n=0;n=(i=a[n]).left&&t<=i.left+i.width&&e>=i.top&&e<=i.top+i.height)return r.legendItems[n];return null},handleEvent:function(t){var e,n=this,i=n.options,a="mouseup"===t.type?"click":t.type;if("mousemove"===a){if(!i.onHover&&!i.onLeave)return}else{if("click"!==a)return;if(!i.onClick)return}e=n._getLegendItemAt(t.x,t.y),"click"===a?e&&i.onClick&&i.onClick.call(n,t.native,e):(i.onLeave&&e!==n._hoveredItem&&(n._hoveredItem&&i.onLeave.call(n,t.native,n._hoveredItem),n._hoveredItem=e),i.onHover&&e&&i.onHover.call(n,t.native,e))}});function Oi(t,e){var n=new Ti({ctx:t.ctx,options:e,chart:t});me.configure(t,n,e),me.addBox(t,n),t.legend=n}var Ai={id:"legend",_element:Ti,beforeInit:function(t){var e=t.options.legend;e&&Oi(t,e)},beforeUpdate:function(t){var e=t.options.legend,n=t.legend;e?(H.mergeIf(e,W.global.legend),n?(me.configure(t,n,e),n.options=e):Oi(t,e)):n&&(me.removeBox(t,n),delete t.legend)},afterEvent:function(t,e){var n=t.legend;n&&n.handleEvent(e)}},Fi=H.noop;W._set("global",{title:{display:!1,fontStyle:"bold",fullWidth:!0,padding:10,position:"top",text:"",weight:2e3}});var Ii=$.extend({initialize:function(t){H.extend(this,t),this.legendHitBoxes=[]},beforeUpdate:Fi,update:function(t,e,n){var i=this;return i.beforeUpdate(),i.maxWidth=t,i.maxHeight=e,i.margins=n,i.beforeSetDimensions(),i.setDimensions(),i.afterSetDimensions(),i.beforeBuildLabels(),i.buildLabels(),i.afterBuildLabels(),i.beforeFit(),i.fit(),i.afterFit(),i.afterUpdate(),i.minSize},afterUpdate:Fi,beforeSetDimensions:Fi,setDimensions:function(){var t=this;t.isHorizontal()?(t.width=t.maxWidth,t.left=0,t.right=t.width):(t.height=t.maxHeight,t.top=0,t.bottom=t.height),t.paddingLeft=0,t.paddingTop=0,t.paddingRight=0,t.paddingBottom=0,t.minSize={width:0,height:0}},afterSetDimensions:Fi,beforeBuildLabels:Fi,buildLabels:Fi,afterBuildLabels:Fi,beforeFit:Fi,fit:function(){var t,e=this,n=e.options,i=e.minSize={},a=e.isHorizontal();n.display?(t=(H.isArray(n.text)?n.text.length:1)*H.options._parseFont(n).lineHeight+2*n.padding,e.width=i.width=a?e.maxWidth:t,e.height=i.height=a?t:e.maxHeight):e.width=i.width=e.height=i.height=0},afterFit:Fi,isHorizontal:function(){var t=this.options.position;return"top"===t||"bottom"===t},draw:function(){var t=this,e=t.ctx,n=t.options;if(n.display){var i,a,r,o=H.options._parseFont(n),s=o.lineHeight,l=s/2+n.padding,u=0,d=t.top,h=t.left,c=t.bottom,f=t.right;e.fillStyle=H.valueOrDefault(n.fontColor,W.global.defaultFontColor),e.font=o.string,t.isHorizontal()?(a=h+(f-h)/2,r=d+l,i=f-h):(a="left"===n.position?h+l:f-l,r=d+(c-d)/2,i=c-d,u=Math.PI*("left"===n.position?-.5:.5)),e.save(),e.translate(a,r),e.rotate(u),e.textAlign="center",e.textBaseline="middle";var g=n.text;if(H.isArray(g))for(var m=0,p=0;p=0;i--){var a=t[i];if(e(a))return a}},H.isNumber=function(t){return!isNaN(parseFloat(t))&&isFinite(t)},H.almostEquals=function(t,e,n){return Math.abs(t-e)=t},H.max=function(t){return t.reduce((function(t,e){return isNaN(e)?t:Math.max(t,e)}),Number.NEGATIVE_INFINITY)},H.min=function(t){return t.reduce((function(t,e){return isNaN(e)?t:Math.min(t,e)}),Number.POSITIVE_INFINITY)},H.sign=Math.sign?function(t){return Math.sign(t)}:function(t){return 0===(t=+t)||isNaN(t)?t:t>0?1:-1},H.toRadians=function(t){return t*(Math.PI/180)},H.toDegrees=function(t){return t*(180/Math.PI)},H._decimalPlaces=function(t){if(H.isFinite(t)){for(var e=1,n=0;Math.round(t*e)/e!==t;)e*=10,n++;return n}},H.getAngleFromPoint=function(t,e){var n=e.x-t.x,i=e.y-t.y,a=Math.sqrt(n*n+i*i),r=Math.atan2(i,n);return r<-.5*Math.PI&&(r+=2*Math.PI),{angle:r,distance:a}},H.distanceBetweenPoints=function(t,e){return Math.sqrt(Math.pow(e.x-t.x,2)+Math.pow(e.y-t.y,2))},H.aliasPixel=function(t){return t%2==0?0:.5},H._alignPixel=function(t,e,n){var i=t.currentDevicePixelRatio,a=n/2;return Math.round((e-a)*i)/i+a},H.splineCurve=function(t,e,n,i){var a=t.skip?e:t,r=e,o=n.skip?e:n,s=Math.sqrt(Math.pow(r.x-a.x,2)+Math.pow(r.y-a.y,2)),l=Math.sqrt(Math.pow(o.x-r.x,2)+Math.pow(o.y-r.y,2)),u=s/(s+l),d=l/(s+l),h=i*(u=isNaN(u)?0:u),c=i*(d=isNaN(d)?0:d);return{previous:{x:r.x-h*(o.x-a.x),y:r.y-h*(o.y-a.y)},next:{x:r.x+c*(o.x-a.x),y:r.y+c*(o.y-a.y)}}},H.EPSILON=Number.EPSILON||1e-14,H.splineCurveMonotone=function(t){var e,n,i,a,r,o,s,l,u,d=(t||[]).map((function(t){return{model:t._model,deltaK:0,mK:0}})),h=d.length;for(e=0;e0?d[e-1]:null,(a=e0?d[e-1]:null,a=e=t.length-1?t[0]:t[e+1]:e>=t.length-1?t[t.length-1]:t[e+1]},H.previousItem=function(t,e,n){return n?e<=0?t[t.length-1]:t[e-1]:e<=0?t[0]:t[e-1]},H.niceNum=function(t,e){var n=Math.floor(H.log10(t)),i=t/Math.pow(10,n);return(e?i<1.5?1:i<3?2:i<7?5:10:i<=1?1:i<=2?2:i<=5?5:10)*Math.pow(10,n)},H.requestAnimFrame="undefined"==typeof window?function(t){t()}:window.requestAnimationFrame||window.webkitRequestAnimationFrame||window.mozRequestAnimationFrame||window.oRequestAnimationFrame||window.msRequestAnimationFrame||function(t){return window.setTimeout(t,1e3/60)},H.getRelativePosition=function(t,e){var n,i,a=t.originalEvent||t,r=t.target||t.srcElement,o=r.getBoundingClientRect(),s=a.touches;s&&s.length>0?(n=s[0].clientX,i=s[0].clientY):(n=a.clientX,i=a.clientY);var l=parseFloat(H.getStyle(r,"padding-left")),u=parseFloat(H.getStyle(r,"padding-top")),d=parseFloat(H.getStyle(r,"padding-right")),h=parseFloat(H.getStyle(r,"padding-bottom")),c=o.right-o.left-l-d,f=o.bottom-o.top-u-h;return{x:n=Math.round((n-o.left-l)/c*r.width/e.currentDevicePixelRatio),y:i=Math.round((i-o.top-u)/f*r.height/e.currentDevicePixelRatio)}},H.getConstraintWidth=function(t){return n(t,"max-width","clientWidth")},H.getConstraintHeight=function(t){return n(t,"max-height","clientHeight")},H._calculatePadding=function(t,e,n){return(e=H.getStyle(t,e)).indexOf("%")>-1?n*parseInt(e,10)/100:parseInt(e,10)},H._getParentNode=function(t){var e=t.parentNode;return e&&"[object ShadowRoot]"===e.toString()&&(e=e.host),e},H.getMaximumWidth=function(t){var e=H._getParentNode(t);if(!e)return t.clientWidth;var n=e.clientWidth,i=n-H._calculatePadding(e,"padding-left",n)-H._calculatePadding(e,"padding-right",n),a=H.getConstraintWidth(t);return isNaN(a)?i:Math.min(i,a)},H.getMaximumHeight=function(t){var e=H._getParentNode(t);if(!e)return t.clientHeight;var n=e.clientHeight,i=n-H._calculatePadding(e,"padding-top",n)-H._calculatePadding(e,"padding-bottom",n),a=H.getConstraintHeight(t);return isNaN(a)?i:Math.min(i,a)},H.getStyle=function(t,e){return t.currentStyle?t.currentStyle[e]:document.defaultView.getComputedStyle(t,null).getPropertyValue(e)},H.retinaScale=function(t,e){var n=t.currentDevicePixelRatio=e||"undefined"!=typeof window&&window.devicePixelRatio||1;if(1!==n){var i=t.canvas,a=t.height,r=t.width;i.height=a*n,i.width=r*n,t.ctx.scale(n,n),i.style.height||i.style.width||(i.style.height=a+"px",i.style.width=r+"px")}},H.fontString=function(t,e,n){return e+" "+t+"px "+n},H.longestText=function(t,e,n,i){var a=(i=i||{}).data=i.data||{},r=i.garbageCollect=i.garbageCollect||[];i.font!==e&&(a=i.data={},r=i.garbageCollect=[],i.font=e),t.font=e;var o,s,l,u,d,h=0,c=n.length;for(o=0;on.length){for(o=0;oi&&(i=r),i},H.numberOfLabelLines=function(t){var e=1;return H.each(t,(function(t){H.isArray(t)&&t.length>e&&(e=t.length)})),e},H.color=k?function(t){return t instanceof CanvasGradient&&(t=W.global.defaultColor),k(t)}:function(t){return console.error("Color.js not found!"),t},H.getHoverColor=function(t){return t instanceof CanvasPattern||t instanceof CanvasGradient?t:H.color(t).saturate(.5).darken(.1).rgbString()}}(),en._adapters=rn,en.Animation=K,en.animationService=J,en.controllers=Jt,en.DatasetController=it,en.defaults=W,en.Element=$,en.elements=wt,en.Interaction=re,en.layouts=me,en.platform=Ie,en.plugins=Le,en.Scale=xn,en.scaleService=Re,en.Ticks=on,en.Tooltip=Ge,en.helpers.each(fi,(function(t,e){en.scaleService.registerScaleType(e,t,t._defaults)})),Ri)Ri.hasOwnProperty(zi)&&en.plugins.register(Ri[zi]);en.platform.initialize();var Ei=en;return"undefined"!=typeof window&&(window.Chart=en),en.Chart=en,en.Legend=Ri.legend._element,en.Title=Ri.title._element,en.pluginService=en.plugins,en.PluginBase=en.Element.extend({}),en.canvasHelpers=en.helpers.canvas,en.layoutService=en.layouts,en.LinearScaleBase=Dn,en.helpers.each(["Bar","Bubble","Doughnut","Line","PolarArea","Radar","Scatter"],(function(t){en[t]=function(e,n){return new en(e,en.helpers.merge(n||{},{type:t.charAt(0).toLowerCase()+t.slice(1)}))}})),Ei})); diff --git a/src/manifest.json b/src/manifest.json index 59d0a44..fdc2682 100644 --- a/src/manifest.json +++ b/src/manifest.json @@ -17,6 +17,7 @@ "js": [ "components/helpers.js", "components/module_parser.js", + "components/Chart.bundle.min.js", "main.js"], "css": ["components/custom_styles.css"] } From 2ca0d782a87df4a683a3573316cd9ca776cd7e44 Mon Sep 17 00:00:00 2001 From: Letum Date: Wed, 22 Jan 2020 14:43:26 +0100 Subject: [PATCH 02/36] add a fancy graph for your term credits --- src/components/module_parser.js | 46 ++++++++++++++++++ src/main.js | 86 +++++++++++++++++++++++++++++++-- 2 files changed, 129 insertions(+), 3 deletions(-) diff --git a/src/components/module_parser.js b/src/components/module_parser.js index 3d0c34e..219e010 100644 --- a/src/components/module_parser.js +++ b/src/components/module_parser.js @@ -76,6 +76,8 @@ const ModuleParser = { const modules = []; + + let moduleTypeList = await fetch(Helpers.getExtensionInternalFileUrl('data/modules_i.json')) .then(response => response.json()); @@ -92,6 +94,15 @@ const ModuleParser = { return; } + let anlassListe = anlasslistApiResponse.items; + const firstModul = anlassListe[anlassListe.length - 1]; + + const STARTMONTHSPRINGTERM = 1 + const STARTMONTHAUTUMNTERM = 8 + + const firstModulYear = (new Date(firstModul.from)).getFullYear(); + const firstModulMonth = (new Date(firstModul.from)).getMonth(); + anlasslistApiResponse.items.forEach(item => { let parsedModule = {}; @@ -101,6 +112,41 @@ const ModuleParser = { parsedModule[MarkKey] = item.note === null ? 'n/a' : item.note; parsedModule[GradeKey] = item.grade === null ? 'n/a' : item.grade; + //parsedModule[FromKey] = item.from === null ? 'n/a' : item.from; + + startYear = (new Date(item.from)).getFullYear(); + startMonth = (new Date(item.from)).getMonth(); + + let term = "Something went wrong" + + + if (firstModulMonth == STARTMONTHAUTUMNTERM) { + if (startMonth == STARTMONTHAUTUMNTERM) { + term = (startYear - firstModulYear) * 2 + 1 + } + else if(startMonth == STARTMONTHSPRINGTERM){ + term = (startYear - firstModulYear) * 2 + } + else { + term = "Something went wrong" + } + } + else if (firstModulMonth == STARTMONTHSPRINGTERM) { + if (startMonth == STARTMONTHSPRINGTERM) { + term = (startYear - firstModulYear) * 2 + 1 + } + else if(startMonth == STARTMONTHAUTUMNTERM) { + term = (startYear - firstModulYear) * 2 + 2 + } + else { + term = "Something went wrong" + } + } + else { + term = "Something went wrong" + } + + parsedModule[TermKey] = term; let details = item.details; ItemDetailKeys.forEach(key => { diff --git a/src/main.js b/src/main.js index 5ba9daa..01bb52c 100644 --- a/src/main.js +++ b/src/main.js @@ -2,10 +2,16 @@ const NameKey = 'Nummer'; const CreditsKey = 'ECTS-Punkte'; const MarkKey = 'Bewertung'; const GradeKey = 'Grad'; +const FromKey = 'From' +const TermKey = 'Term' const ItemDetailKeys = [NameKey, CreditsKey]; const ModuleTypeKey = 'Modul-Typ'; const ModuleTableHeaders = [NameKey, ModuleTypeKey, CreditsKey, MarkKey, GradeKey] const GradesCount = { A: 0, B: 0, C: 0, D: 0, E: 0, F: 0 }; + +//const CreditByTermCount = { 1: 0, 2: 0, 3: 0, 4: 0, 5: 0, 6: 0, 7: 0, 8: 0 }; +const CreditByTermCount = [0,0,0,0,0,0,0,0]; + const CreditsByModuleTypeCount = { Kernmodul: { current: 0, min: 66 }, Projektmodul: { current: 0, min: 36 }, @@ -244,11 +250,85 @@ function calculateStats(modules) { }); } +function createChart(div, modules) { + canvas = document.createElement("canvas"); + canvas.setAttribute("id", "myChart"); + div.insertBefore(canvas, div.firstChild); + + + const labels = [] + //CreditByTermCount(Number(modul[TermKey])) += Number(modul[CreditsKey]), + modules.forEach(modul => { + CreditByTermCount[modul[TermKey]] += Number(modul[CreditsKey]); + }) + + for (let index = 0; index < CreditByTermCount.length; index++) { + console.log("CreditByTermCount[index]", CreditByTermCount[index]) + if(CreditByTermCount[index] != 0 ){ + labels.push(index) + } + } + console.log("CreditByTermCount", CreditByTermCount); + + console.log("labels", labels); + + let type = 'line' + let data = { + labels: labels, + datasets: [{ + label: 'Term', + data: [ + CreditByTermCount[1], + CreditByTermCount[2], + CreditByTermCount[3], + CreditByTermCount[4], + CreditByTermCount[5], + CreditByTermCount[6], + CreditByTermCount[7], + CreditByTermCount[8] + ], + backgroundColor: [ + 'rgba(135, 206, 235, 0.5)', + ], + borderColor: [ + 'rgba(135, 206, 235, 1)', + 'rgba(135, 206, 235, 1)', + 'rgba(135, 206, 235, 1)', + 'rgba(135, 206, 235, 1)', + 'rgba(135, 206, 235, 1)', + 'rgba(135, 206, 235, 1)', + 'rgba(135, 206, 235, 1)', + 'rgba(135, 206, 235, 1)' + ], + borderWidth: 2 + }] + } + let options = { + scales: { + yAxes: [{ + ticks: { + beginAtZero: true + } + }] + } + } + + new Chart(canvas, { + type: type, + data: data, + options: options + }); +} + async function generateHtml(modules) { // remove clutter - document.getElementById('intro').remove(); - document.getElementsByClassName('sidebar medium-7 columns mobile-column')[0].remove(); + try { + document.getElementById('intro').remove(); + document.getElementsByClassName('sidebar medium-7 columns mobile-column')[0].remove(); + } catch (error) { + console.log("NOTITLE, DONT CARE") + } let div = document.getElementsByClassName('row teaser-section None')[0]; if (!modules) { @@ -263,7 +343,6 @@ async function generateHtml(modules) { } calculateStats(modules); - createModulesTable(div, modules); Helpers.addTitleToDocument(div, 'Modulübersicht'); @@ -273,6 +352,7 @@ async function generateHtml(modules) { await createGradesOverviewTable(div); createAverageMarkTitle(div); + createChart(div, modules); createStudyTitle(div); } From 0cec0e096152f173438bce26fc563fef600cebe1 Mon Sep 17 00:00:00 2001 From: Letum Date: Wed, 22 Jan 2020 17:39:12 +0100 Subject: [PATCH 03/36] fix if first modul is earlier then normal --- src/components/module_parser.js | 22 +++++++++++++--------- 1 file changed, 13 insertions(+), 9 deletions(-) diff --git a/src/components/module_parser.js b/src/components/module_parser.js index 219e010..8541f14 100644 --- a/src/components/module_parser.js +++ b/src/components/module_parser.js @@ -97,8 +97,12 @@ const ModuleParser = { let anlassListe = anlasslistApiResponse.items; const firstModul = anlassListe[anlassListe.length - 1]; - const STARTMONTHSPRINGTERM = 1 - const STARTMONTHAUTUMNTERM = 8 + const STARTMONTHSPRINGTERM = 0 + const STARTMONTHAUTUMNTERM = 5 + + const ENDMONTHAUTUMNTERM = 11 + const ENDMONTHSPRINGTERM = 4 + const firstModulYear = (new Date(firstModul.from)).getFullYear(); const firstModulMonth = (new Date(firstModul.from)).getMonth(); @@ -112,30 +116,30 @@ const ModuleParser = { parsedModule[MarkKey] = item.note === null ? 'n/a' : item.note; parsedModule[GradeKey] = item.grade === null ? 'n/a' : item.grade; - //parsedModule[FromKey] = item.from === null ? 'n/a' : item.from; startYear = (new Date(item.from)).getFullYear(); startMonth = (new Date(item.from)).getMonth(); + endMonth = (new Date(item.to)).getMonth(); let term = "Something went wrong" - if (firstModulMonth == STARTMONTHAUTUMNTERM) { - if (startMonth == STARTMONTHAUTUMNTERM) { + if (firstModulMonth >= STARTMONTHAUTUMNTERM) { + if (endMonth == ENDMONTHAUTUMNTERM) { term = (startYear - firstModulYear) * 2 + 1 } - else if(startMonth == STARTMONTHSPRINGTERM){ + else if(endMonth >= ENDMONTHSPRINGTERM){ term = (startYear - firstModulYear) * 2 } else { term = "Something went wrong" } } - else if (firstModulMonth == STARTMONTHSPRINGTERM) { - if (startMonth == STARTMONTHSPRINGTERM) { + else if (firstModulMonth >= STARTMONTHSPRINGTERM) { + if (endMonth == ENDMONTHSPRINGTERM) { term = (startYear - firstModulYear) * 2 + 1 } - else if(startMonth == STARTMONTHAUTUMNTERM) { + else if(starendMonthtMonth == ENDMONTHAUTUMNTERM) { term = (startYear - firstModulYear) * 2 + 2 } else { From 8ebe84423a9beb7700b769306aba3cc84ff8477e Mon Sep 17 00:00:00 2001 From: eddex Date: Wed, 22 Jan 2020 17:56:07 +0100 Subject: [PATCH 04/36] move files and rename 'Term' to 'Semester' --- src/{components => lib}/Chart.bundle.min.js | 0 src/main.js | 34 +++++++++---------- src/manifest.json | 4 +-- .../credits_by_module_type_table.html | 0 .../custom_styles.css | 0 .../grades_table.html | 0 6 files changed, 19 insertions(+), 19 deletions(-) rename src/{components => lib}/Chart.bundle.min.js (100%) rename src/{components => templates}/credits_by_module_type_table.html (100%) rename src/{components => templates}/custom_styles.css (100%) rename src/{components => templates}/grades_table.html (100%) diff --git a/src/components/Chart.bundle.min.js b/src/lib/Chart.bundle.min.js similarity index 100% rename from src/components/Chart.bundle.min.js rename to src/lib/Chart.bundle.min.js diff --git a/src/main.js b/src/main.js index 01bb52c..a4db087 100644 --- a/src/main.js +++ b/src/main.js @@ -10,7 +10,7 @@ const ModuleTableHeaders = [NameKey, ModuleTypeKey, CreditsKey, MarkKey, GradeKe const GradesCount = { A: 0, B: 0, C: 0, D: 0, E: 0, F: 0 }; //const CreditByTermCount = { 1: 0, 2: 0, 3: 0, 4: 0, 5: 0, 6: 0, 7: 0, 8: 0 }; -const CreditByTermCount = [0,0,0,0,0,0,0,0]; +const CreditsBySemesterCount = [0,0,0,0,0,0,0,0]; const CreditsByModuleTypeCount = { Kernmodul: { current: 0, min: 66 }, @@ -120,7 +120,7 @@ function createModulesTable(div, modules) { */ async function createCreditsByModuleTypeTable(div) { - let template = await fetch(Helpers.getExtensionInternalFileUrl('components/credits_by_module_type_table.html')) + let template = await fetch(Helpers.getExtensionInternalFileUrl('templates/credits_by_module_type_table.html')) .then(response => response.text()); let creditsByModuleTypeTable = document.createElement('div'); creditsByModuleTypeTable.innerHTML = template; @@ -146,7 +146,7 @@ async function createCreditsByModuleTypeTable(div) { */ async function createGradesOverviewTable(div) { - let gradesTableTemplate = await fetch(Helpers.getExtensionInternalFileUrl('components/grades_table.html')) + let gradesTableTemplate = await fetch(Helpers.getExtensionInternalFileUrl('templates/grades_table.html')) .then(response => response.text()); let gradeOverviewTable = document.createElement('div'); @@ -259,16 +259,16 @@ function createChart(div, modules) { const labels = [] //CreditByTermCount(Number(modul[TermKey])) += Number(modul[CreditsKey]), modules.forEach(modul => { - CreditByTermCount[modul[TermKey]] += Number(modul[CreditsKey]); + CreditsBySemesterCount[modul[TermKey]] += Number(modul[CreditsKey]); }) - for (let index = 0; index < CreditByTermCount.length; index++) { - console.log("CreditByTermCount[index]", CreditByTermCount[index]) - if(CreditByTermCount[index] != 0 ){ + for (let index = 0; index < CreditsBySemesterCount.length; index++) { + console.log("CreditBySemesterCount[index]", CreditsBySemesterCount[index]) + if(CreditsBySemesterCount[index] != 0 ){ labels.push(index) } } - console.log("CreditByTermCount", CreditByTermCount); + console.log("CreditBySemesterCount", CreditsBySemesterCount); console.log("labels", labels); @@ -276,16 +276,16 @@ function createChart(div, modules) { let data = { labels: labels, datasets: [{ - label: 'Term', + label: 'Semester', data: [ - CreditByTermCount[1], - CreditByTermCount[2], - CreditByTermCount[3], - CreditByTermCount[4], - CreditByTermCount[5], - CreditByTermCount[6], - CreditByTermCount[7], - CreditByTermCount[8] + CreditsBySemesterCount[1], + CreditsBySemesterCount[2], + CreditsBySemesterCount[3], + CreditsBySemesterCount[4], + CreditsBySemesterCount[5], + CreditsBySemesterCount[6], + CreditsBySemesterCount[7], + CreditsBySemesterCount[8] ], backgroundColor: [ 'rgba(135, 206, 235, 0.5)', diff --git a/src/manifest.json b/src/manifest.json index fdc2682..3241724 100644 --- a/src/manifest.json +++ b/src/manifest.json @@ -17,9 +17,9 @@ "js": [ "components/helpers.js", "components/module_parser.js", - "components/Chart.bundle.min.js", + "lib/Chart.bundle.min.js", "main.js"], - "css": ["components/custom_styles.css"] + "css": ["templates/custom_styles.css"] } ], diff --git a/src/components/credits_by_module_type_table.html b/src/templates/credits_by_module_type_table.html similarity index 100% rename from src/components/credits_by_module_type_table.html rename to src/templates/credits_by_module_type_table.html diff --git a/src/components/custom_styles.css b/src/templates/custom_styles.css similarity index 100% rename from src/components/custom_styles.css rename to src/templates/custom_styles.css diff --git a/src/components/grades_table.html b/src/templates/grades_table.html similarity index 100% rename from src/components/grades_table.html rename to src/templates/grades_table.html From b749496ff6a90c3f6826c8fe28e722cf852961f4 Mon Sep 17 00:00:00 2001 From: eddex Date: Wed, 22 Jan 2020 18:05:11 +0100 Subject: [PATCH 05/36] update readme for new folder structure --- README.md | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/README.md b/README.md index f0f1e4d..ba64a30 100644 --- a/README.md +++ b/README.md @@ -46,9 +46,11 @@ Chrome does not support third party extensions anymore. Maybe we'll publish the ## Create a new release 1. add everthing from `src` to a `.zip` file: + - `components/` directory - `data/` directory - `icons/` directory - - `components/` directory + - `lib/` directory + - `templates/` directory - `LICENSE` file - `main.js` file - `manifest.json` file From d7bbd27cd9135e86aa70d26d989349ff06244a62 Mon Sep 17 00:00:00 2001 From: Letum Date: Wed, 22 Jan 2020 18:06:57 +0100 Subject: [PATCH 06/36] fix semester credit lookup --- src/components/module_parser.js | 25 ++++++++++++++----------- 1 file changed, 14 insertions(+), 11 deletions(-) diff --git a/src/components/module_parser.js b/src/components/module_parser.js index 8541f14..6f4d9b9 100644 --- a/src/components/module_parser.js +++ b/src/components/module_parser.js @@ -121,36 +121,39 @@ const ModuleParser = { startMonth = (new Date(item.from)).getMonth(); endMonth = (new Date(item.to)).getMonth(); - let term = "Something went wrong" + let semester = "n/a" if (firstModulMonth >= STARTMONTHAUTUMNTERM) { - if (endMonth == ENDMONTHAUTUMNTERM) { - term = (startYear - firstModulYear) * 2 + 1 + if (endMonth < 2 || endMonth > 10) { + semester = (startYear - firstModulYear) * 2 + 1 } - else if(endMonth >= ENDMONTHSPRINGTERM){ - term = (startYear - firstModulYear) * 2 + else if(endMonth > 3 && endMonth < 11){ + semester = (startYear - firstModulYear) * 2 } else { - term = "Something went wrong" + semester = "n/a" } } else if (firstModulMonth >= STARTMONTHSPRINGTERM) { if (endMonth == ENDMONTHSPRINGTERM) { - term = (startYear - firstModulYear) * 2 + 1 + semester = (startYear - firstModulYear) * 2 + 1 } else if(starendMonthtMonth == ENDMONTHAUTUMNTERM) { - term = (startYear - firstModulYear) * 2 + 2 + semester = (startYear - firstModulYear) * 2 + 2 } else { - term = "Something went wrong" + semester = "n/a" } } else { - term = "Something went wrong" + semester = "n/a" } - parsedModule[TermKey] = term; + parsedModule[TermKey] = semester; + + console.log(parsedModule) + console.log(endMonth) let details = item.details; ItemDetailKeys.forEach(key => { From d02d0b4f80f69c3c9246141d8df6f78364071e54 Mon Sep 17 00:00:00 2001 From: eddex Date: Wed, 22 Jan 2020 18:41:13 +0100 Subject: [PATCH 07/36] minor refactoring --- src/components/module_parser.js | 35 ++++++++++++++------------------- src/main.js | 5 +---- 2 files changed, 16 insertions(+), 24 deletions(-) diff --git a/src/components/module_parser.js b/src/components/module_parser.js index 8541f14..8b23b92 100644 --- a/src/components/module_parser.js +++ b/src/components/module_parser.js @@ -76,8 +76,6 @@ const ModuleParser = { const modules = []; - - let moduleTypeList = await fetch(Helpers.getExtensionInternalFileUrl('data/modules_i.json')) .then(response => response.json()); @@ -94,8 +92,7 @@ const ModuleParser = { return; } - let anlassListe = anlasslistApiResponse.items; - const firstModul = anlassListe[anlassListe.length - 1]; + const firstModule = anlasslistApiResponse.items[anlasslistApiResponse.items.length - 1]; const STARTMONTHSPRINGTERM = 0 const STARTMONTHAUTUMNTERM = 5 @@ -103,9 +100,8 @@ const ModuleParser = { const ENDMONTHAUTUMNTERM = 11 const ENDMONTHSPRINGTERM = 4 - - const firstModulYear = (new Date(firstModul.from)).getFullYear(); - const firstModulMonth = (new Date(firstModul.from)).getMonth(); + const firstModuleYear = (new Date(firstModule.from)).getFullYear(); + const firstModuleMonth = (new Date(firstModule.from)).getMonth(); anlasslistApiResponse.items.forEach(item => { @@ -121,36 +117,35 @@ const ModuleParser = { startMonth = (new Date(item.from)).getMonth(); endMonth = (new Date(item.to)).getMonth(); - let term = "Something went wrong" + let semester = "Something went wrong" - - if (firstModulMonth >= STARTMONTHAUTUMNTERM) { + if (firstModuleMonth >= STARTMONTHAUTUMNTERM) { if (endMonth == ENDMONTHAUTUMNTERM) { - term = (startYear - firstModulYear) * 2 + 1 + semester = (startYear - firstModuleYear) * 2 + 1 } else if(endMonth >= ENDMONTHSPRINGTERM){ - term = (startYear - firstModulYear) * 2 + semester = (startYear - firstModuleYear) * 2 } else { - term = "Something went wrong" + semester = "Something went wrong" } - } - else if (firstModulMonth >= STARTMONTHSPRINGTERM) { + } + else if (firstModuleMonth >= STARTMONTHSPRINGTERM) { if (endMonth == ENDMONTHSPRINGTERM) { - term = (startYear - firstModulYear) * 2 + 1 + semester = (startYear - firstModuleYear) * 2 + 1 } else if(starendMonthtMonth == ENDMONTHAUTUMNTERM) { - term = (startYear - firstModulYear) * 2 + 2 + semester = (startYear - firstModuleYear) * 2 + 2 } else { - term = "Something went wrong" + semester = "Something went wrong" } } else { - term = "Something went wrong" + semester = "Something went wrong" } - parsedModule[TermKey] = term; + parsedModule.semester = semester; let details = item.details; ItemDetailKeys.forEach(key => { diff --git a/src/main.js b/src/main.js index a4db087..2c3ae81 100644 --- a/src/main.js +++ b/src/main.js @@ -2,8 +2,6 @@ const NameKey = 'Nummer'; const CreditsKey = 'ECTS-Punkte'; const MarkKey = 'Bewertung'; const GradeKey = 'Grad'; -const FromKey = 'From' -const TermKey = 'Term' const ItemDetailKeys = [NameKey, CreditsKey]; const ModuleTypeKey = 'Modul-Typ'; const ModuleTableHeaders = [NameKey, ModuleTypeKey, CreditsKey, MarkKey, GradeKey] @@ -257,9 +255,8 @@ function createChart(div, modules) { const labels = [] - //CreditByTermCount(Number(modul[TermKey])) += Number(modul[CreditsKey]), modules.forEach(modul => { - CreditsBySemesterCount[modul[TermKey]] += Number(modul[CreditsKey]); + CreditsBySemesterCount[modul.semester] += Number(modul[CreditsKey]); }) for (let index = 0; index < CreditsBySemesterCount.length; index++) { From 3a9671493cb89c3561a371da25bee43cef7507a6 Mon Sep 17 00:00:00 2001 From: eddex Date: Wed, 22 Jan 2020 22:17:20 +0100 Subject: [PATCH 08/36] feat: the great burndown chart pls clean up lol --- src/components/module_parser.js | 106 ++++++++++++++++-------------- src/main.js | 112 +++++++++++++++++++++++++++----- 2 files changed, 152 insertions(+), 66 deletions(-) diff --git a/src/components/module_parser.js b/src/components/module_parser.js index de283e5..eb993cf 100644 --- a/src/components/module_parser.js +++ b/src/components/module_parser.js @@ -28,6 +28,57 @@ const ModuleParser = { return ''; }, + /* + * Check if a module was done in Autumn. + * Modules are marked with 'H' for 'Herbstsemester' (autumn) + * or 'F' for 'Frühlingssemester' (spring). + */ + isAutumnSemester: (hsluModule) => { + const includesH = hsluModule.anlassnumber.split('.')[2].includes('H'); + const includesF = hsluModule.anlassnumber.split('.')[2].includes('F'); + return includesH || includesF ? includesH : undefined; + }, + + /* + * Calculate the semester. + */ + calculateSemester: (hsluModule, firstModule) => { + + const startYear = new Date(firstModule.from).getFullYear(); + const isStartInAutumn = ModuleParser.isAutumnSemester(firstModule); + + // the lastPart of the anlassnumber is something like 'F1901' + const lastPart = hsluModule.anlassnumber.split('.')[2]; + const moduleYear = Number('20' + lastPart.substring(1, 3)); + const isModuleInAutumn = ModuleParser.isAutumnSemester(hsluModule); + + const yearDifference = (moduleYear - startYear) + + let semester = undefined; + if (isModuleInAutumn === undefined) { + // we need this early abort because of entries like 'I.BA_PTA_b.1618' + // (which is not even a real module!) + return semester; + } + + if (isStartInAutumn) { + if (isModuleInAutumn) { + semester = yearDifference * 2 + 1; + } + else { + semester = yearDifference * 2; + } + } else { + if (isModuleInAutumn){ + semester = yearDifference * 2 + 2; + } + else { + semester = yearDifference * 2 + 1; + } + } + return semester; + }, + /* * The module name includes the module id. * But there are different formats for the module names. @@ -94,15 +145,6 @@ const ModuleParser = { const firstModule = anlasslistApiResponse.items[anlasslistApiResponse.items.length - 1]; - const STARTMONTHSPRINGTERM = 0 - const STARTMONTHAUTUMNTERM = 5 - - const ENDMONTHAUTUMNTERM = 11 - const ENDMONTHSPRINGTERM = 4 - - const firstModuleYear = (new Date(firstModule.from)).getFullYear(); - const firstModuleMonth = (new Date(firstModule.from)).getMonth(); - anlasslistApiResponse.items.forEach(item => { let parsedModule = {}; @@ -112,50 +154,14 @@ const ModuleParser = { parsedModule[MarkKey] = item.note === null ? 'n/a' : item.note; parsedModule[GradeKey] = item.grade === null ? 'n/a' : item.grade; + parsedModule[NameKey] = item.anlassnumber; + parsedModule.from = item.from; + parsedModule.to = item.to; - startYear = (new Date(item.from)).getFullYear(); - startMonth = (new Date(item.from)).getMonth(); - endMonth = (new Date(item.to)).getMonth(); - - let semester = "n/a" - - if (firstModuleMonth >= STARTMONTHAUTUMNTERM) { - if (endMonth < 2 || endMonth > 10) { - semester = (startYear - firstModuleYear) * 2 + 1 - } - else if(endMonth > 3 && endMonth < 11){ - semester = (startYear - firstModuleYear) * 2 - } - else { - semester = "n/a" - - } - } - else if (firstModuleMonth >= STARTMONTHSPRINGTERM) { - if (endMonth == ENDMONTHSPRINGTERM) { - semester = (startYear - firstModuleYear) * 2 + 1 - } - else if(starendMonthtMonth == ENDMONTHAUTUMNTERM) { - semester = (startYear - firstModuleYear) * 2 + 2 - } - else { - semester = "n/a" - } - } - else { - semester = "n/a" - } - - parsedModule.semester = semester; - - console.log(parsedModule) - console.log(endMonth) + parsedModule.semester = ModuleParser.calculateSemester(item, firstModule); let details = item.details; - ItemDetailKeys.forEach(key => { - let value = ModuleParser.getItemDetailsValueByKey(details, key); - parsedModule[key] = value; - }); + parsedModule[CreditsKey] = ModuleParser.getItemDetailsValueByKey(details, CreditsKey); let moduleId = ModuleParser.getModuleIdFromModuleName(parsedModule[NameKey]); parsedModule[ModuleTypeKey] = moduleTypeList[moduleId]; diff --git a/src/main.js b/src/main.js index 2c3ae81..7c4f90a 100644 --- a/src/main.js +++ b/src/main.js @@ -1,8 +1,7 @@ -const NameKey = 'Nummer'; +const NameKey = 'Modul-Name'; const CreditsKey = 'ECTS-Punkte'; const MarkKey = 'Bewertung'; const GradeKey = 'Grad'; -const ItemDetailKeys = [NameKey, CreditsKey]; const ModuleTypeKey = 'Modul-Typ'; const ModuleTableHeaders = [NameKey, ModuleTypeKey, CreditsKey, MarkKey, GradeKey] const GradesCount = { A: 0, B: 0, C: 0, D: 0, E: 0, F: 0 }; @@ -248,6 +247,19 @@ function calculateStats(modules) { }); } +const getBurndownValue = (semester) => { + if (CreditsBySemesterCount[semester-1] === 0) { + // if no credits were achieved, don't show it in the graph + return undefined; + } + + let burndownValue = 180; + for (let index = 0; index < semester; index++) { + burndownValue -= CreditsBySemesterCount[index] + } + return burndownValue; +} + function createChart(div, modules) { canvas = document.createElement("canvas"); canvas.setAttribute("id", "myChart"); @@ -256,14 +268,14 @@ function createChart(div, modules) { const labels = [] modules.forEach(modul => { - CreditsBySemesterCount[modul.semester] += Number(modul[CreditsKey]); + if (modul[GradeKey] != 'F' && modul.semester != undefined) { + CreditsBySemesterCount[modul.semester - 1] += Number(modul[CreditsKey]); + } }) - for (let index = 0; index < CreditsBySemesterCount.length; index++) { + for (let index = 0; index <= CreditsBySemesterCount.length; index++) { console.log("CreditBySemesterCount[index]", CreditsBySemesterCount[index]) - if(CreditsBySemesterCount[index] != 0 ){ - labels.push(index) - } + labels.push(index) } console.log("CreditBySemesterCount", CreditsBySemesterCount); @@ -273,16 +285,17 @@ function createChart(div, modules) { let data = { labels: labels, datasets: [{ - label: 'Semester', + label: 'Your remaining credits', data: [ - CreditsBySemesterCount[1], - CreditsBySemesterCount[2], - CreditsBySemesterCount[3], - CreditsBySemesterCount[4], - CreditsBySemesterCount[5], - CreditsBySemesterCount[6], - CreditsBySemesterCount[7], - CreditsBySemesterCount[8] + getBurndownValue(0), + getBurndownValue(1), + getBurndownValue(2), + getBurndownValue(3), + getBurndownValue(4), + getBurndownValue(5), + getBurndownValue(6), + getBurndownValue(7), + getBurndownValue(8) ], backgroundColor: [ 'rgba(135, 206, 235, 0.5)', @@ -298,13 +311,80 @@ function createChart(div, modules) { 'rgba(135, 206, 235, 1)' ], borderWidth: 2 + }, + { + label: 'Ideal remaining credits (part time)', + data: [ + 180, + 180 / 8 * 7, + 180 / 8 * 6, + 180 / 8 * 5, + 180 / 8 * 4, + 180 / 8 * 3, + 180 / 8 * 2, + 180 / 8 * 1, + 0 + ], + backgroundColor: [ + 'rgba(255, 206, 235, 0.5)', + ], + borderColor: [ + 'rgba(255, 206, 235, 1)', + 'rgba(255, 206, 235, 1)', + 'rgba(255, 206, 235, 1)', + 'rgba(255, 206, 235, 1)', + 'rgba(255, 206, 235, 1)', + 'rgba(255, 206, 235, 1)', + 'rgba(255, 206, 235, 1)', + 'rgba(255, 206, 235, 1)' + ], + borderWidth: 2 + }, + { + label: 'Ideal remaining credits (full time)', + data: [ + 180, + 180 / 6 * 5, + 180 / 6 * 4, + 180 / 6 * 3, + 180 / 6 * 2, + 180 / 6 * 1, + 0 + ], + backgroundColor: [ + 'rgba(206, 255, 235, 0.5)', + ], + borderColor: [ + 'rgba(206, 255, 235, 1)', + 'rgba(206, 255, 235, 1)', + 'rgba(206, 255, 235, 1)', + 'rgba(206, 255, 235, 1)', + 'rgba(206, 255, 235, 1)', + 'rgba(206, 255, 235, 1)', + 'rgba(206, 255, 235, 1)', + 'rgba(206, 255, 235, 1)' + ], + borderWidth: 2 }] } + let options = { scales: { yAxes: [{ ticks: { beginAtZero: true + }, + display: true, + scaleLabel: { + display: true, + labelString: 'Remaining credits' + } + }], + xAxes: [{ + display: true, + scaleLabel: { + display: true, + labelString: 'Semester' } }] } From 707af11b8b27fecb1d81d238c577b3e804e1d6c4 Mon Sep 17 00:00:00 2001 From: eddex Date: Wed, 22 Jan 2020 22:39:26 +0100 Subject: [PATCH 09/36] update readme --- README.md | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/README.md b/README.md index ba64a30..de2af05 100644 --- a/README.md +++ b/README.md @@ -5,8 +5,16 @@ Brought to you by [@Lextum](https://github.com/Lextum) and [@eddex](https://gith Want to add some missing modules, fix a bug or add a new, awesome feature? That's great. But please read [CONTRIBUTING.md](CONTRIBUTING.md) first! +## Features + ![screenshot](screenshot.png) +The browser add-on puts the following features to the page 'Meine Anmeldungen' on 'MyCampus': +- A simple table with all your modules, credits and grades +- An overview of how many modules of each type have been done / are still needed. +- An overview of your grades including grade distribution and average. +- A burndown chart to visualize your credit progress. + ## Installation ### Firefox From 5bdabffd40e57745d1550b390b632db313e52f58 Mon Sep 17 00:00:00 2001 From: Letum Date: Wed, 22 Jan 2020 23:10:10 +0100 Subject: [PATCH 10/36] fix for the case if the first modul can't be fixed --- src/components/module_parser.js | 11 +++++++++-- 1 file changed, 9 insertions(+), 2 deletions(-) diff --git a/src/components/module_parser.js b/src/components/module_parser.js index eb993cf..0fbf815 100644 --- a/src/components/module_parser.js +++ b/src/components/module_parser.js @@ -143,8 +143,15 @@ const ModuleParser = { return; } - const firstModule = anlasslistApiResponse.items[anlasslistApiResponse.items.length - 1]; - + let firstModule = anlasslistApiResponse.items[anlasslistApiResponse.items.length - 1]; + let i = 1; + isAutumnSemester = ModuleParser.isAutumnSemester(firstModule); + + while(isAutumnSemester == undefined){ + firstModule = anlasslistApiResponse.items[anlasslistApiResponse.items.length - i]; + isAutumnSemester = ModuleParser.isAutumnSemester(firstModule); + i++; + } anlasslistApiResponse.items.forEach(item => { let parsedModule = {}; From 4c1d5d50122e913d5d96d6e61476d846806f4285 Mon Sep 17 00:00:00 2001 From: Letum Date: Wed, 22 Jan 2020 23:48:06 +0100 Subject: [PATCH 11/36] fixer upper of findModul --- src/components/module_parser.js | 13 ++++--------- src/main.js | 4 ---- 2 files changed, 4 insertions(+), 13 deletions(-) diff --git a/src/components/module_parser.js b/src/components/module_parser.js index 0fbf815..d9bfcc9 100644 --- a/src/components/module_parser.js +++ b/src/components/module_parser.js @@ -69,7 +69,7 @@ const ModuleParser = { semester = yearDifference * 2; } } else { - if (isModuleInAutumn){ + if (isModuleInAutumn) { semester = yearDifference * 2 + 2; } else { @@ -143,15 +143,10 @@ const ModuleParser = { return; } - let firstModule = anlasslistApiResponse.items[anlasslistApiResponse.items.length - 1]; - let i = 1; - isAutumnSemester = ModuleParser.isAutumnSemester(firstModule); + firstModule = anlasslistApiResponse.items + .reverse() + .find(modul => ModuleParser.isAutumnSemester(modul) != undefined) - while(isAutumnSemester == undefined){ - firstModule = anlasslistApiResponse.items[anlasslistApiResponse.items.length - i]; - isAutumnSemester = ModuleParser.isAutumnSemester(firstModule); - i++; - } anlasslistApiResponse.items.forEach(item => { let parsedModule = {}; diff --git a/src/main.js b/src/main.js index 7c4f90a..159ef7d 100644 --- a/src/main.js +++ b/src/main.js @@ -274,12 +274,8 @@ function createChart(div, modules) { }) for (let index = 0; index <= CreditsBySemesterCount.length; index++) { - console.log("CreditBySemesterCount[index]", CreditsBySemesterCount[index]) labels.push(index) } - console.log("CreditBySemesterCount", CreditsBySemesterCount); - - console.log("labels", labels); let type = 'line' let data = { From 030e3efb689ed80fc03cf952dd7cf4cb874a41b6 Mon Sep 17 00:00:00 2001 From: Letum Date: Thu, 23 Jan 2020 13:43:36 +0100 Subject: [PATCH 12/36] small fix, so array is in the right order --- src/components/module_parser.js | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/components/module_parser.js b/src/components/module_parser.js index d9bfcc9..d99c929 100644 --- a/src/components/module_parser.js +++ b/src/components/module_parser.js @@ -144,8 +144,9 @@ const ModuleParser = { } firstModule = anlasslistApiResponse.items + .slice() .reverse() - .find(modul => ModuleParser.isAutumnSemester(modul) != undefined) + .find(modul => ModuleParser.isAutumnSemester(modul) != undefined); anlasslistApiResponse.items.forEach(item => { From 5877e025846d8040d87ad2a8dbcc52365948a6d2 Mon Sep 17 00:00:00 2001 From: Letum Date: Thu, 23 Jan 2020 21:11:16 +0100 Subject: [PATCH 13/36] add a total credits by semester bar --- src/main.js | 76 +++++++++++++++++++++++++++++++++++++++++++++++------ 1 file changed, 68 insertions(+), 8 deletions(-) diff --git a/src/main.js b/src/main.js index 159ef7d..18cd5b4 100644 --- a/src/main.js +++ b/src/main.js @@ -278,9 +278,41 @@ function createChart(div, modules) { } let type = 'line' + let data = { labels: labels, - datasets: [{ + datasets: [ + { + label: 'Total Credits by Semster', + data: + [ + 0, + CreditsBySemesterCount[0], + CreditsBySemesterCount[1], + CreditsBySemesterCount[2], + CreditsBySemesterCount[3], + CreditsBySemesterCount[4], + CreditsBySemesterCount[5], + CreditsBySemesterCount[6], + CreditsBySemesterCount[7] + ], + + backgroundColor: [ + 'rgba(65, 94, 108, 0.8)', + 'rgba(65, 94, 108, 0.8)', + 'rgba(65, 94, 108, 0.8)', + 'rgba(65, 94, 108, 0.8)', + 'rgba(65, 94, 108, 0.8)', + 'rgba(65, 94, 108, 0.8)', + 'rgba(65, 94, 108, 0.8)', + 'rgba(65, 94, 108, 0.8)', + ], + // Changes this dataset to become a line + type: 'bar', + yAxisID: 'y-axis-2' + + }, + { label: 'Your remaining credits', data: [ getBurndownValue(0), @@ -306,9 +338,14 @@ function createChart(div, modules) { 'rgba(135, 206, 235, 1)', 'rgba(135, 206, 235, 1)' ], - borderWidth: 2 - }, - { + borderWidth: 2, + yAxisID: 'y-axis-1' + + } + ] + } + if (isPartTime) { + data.datasets.push({ label: 'Ideal remaining credits (part time)', data: [ 180, @@ -335,7 +372,10 @@ function createChart(div, modules) { 'rgba(255, 206, 235, 1)' ], borderWidth: 2 - }, + }) + } + else { + data.datasets.push( { label: 'Ideal remaining credits (full time)', data: [ @@ -361,21 +401,41 @@ function createChart(div, modules) { 'rgba(206, 255, 235, 1)' ], borderWidth: 2 - }] + } + ) } let options = { scales: { - yAxes: [{ + yAxes: [ + { ticks: { beginAtZero: true }, display: true, + position: 'left', scaleLabel: { display: true, labelString: 'Remaining credits' + }, + id: 'y-axis-1', + + }, + { + ticks: { + suggestedMin: 0, + suggestedMax: 180 + }, + type: 'linear', // only linear but allow scale type registration. This allows extensions to exist solely for log scale for instance + display: true, + position: 'right', + scaleLabel: { + display: true, + labelString: 'Credits by semster' + }, + id: 'y-axis-2', } - }], + ], xAxes: [{ display: true, scaleLabel: { From 3680651e2b9605211d59adfaef8b8fa5f88a05ad Mon Sep 17 00:00:00 2001 From: Letum Date: Thu, 23 Jan 2020 21:12:05 +0100 Subject: [PATCH 14/36] add function to check if student is parttime --- src/main.js | 23 +++++++++++++++++++++++ 1 file changed, 23 insertions(+) diff --git a/src/main.js b/src/main.js index 18cd5b4..2234f20 100644 --- a/src/main.js +++ b/src/main.js @@ -69,6 +69,29 @@ async function getStudyTitle() { return title; } +/** + * Return if Student is Partime + * @returns {boolean} + * */ +function isPartTimeStudent(studentData) { + + const searchStringStart = '
Ausbildungsform
'; + const searchStringEnd = ""; + + return studentData.split(searchStringStart)[1].split(searchStringEnd)[0].includes('Berufsbegleitend'); +} + +async function getStudentInformations() { + const URL = "https://mycampus.hslu.ch/de-ch/stud-i/mein-studium/meine-daten/" + + let studentData = await fetch((URL)) + .then(response => response.text()); + + studentInformations.isPartTime = isPartTimeStudent(studentData); + + studentInformations.studyTitle = getStudyTitle(studentData); + studentInformations.studyAcronym = getStudyAcronym(studentInformations.studyTitle); +} /* * Creates one row of the modules table. */ From fa373d2a7e0fe52467c85de7199595d238299e35 Mon Sep 17 00:00:00 2001 From: Letum Date: Thu, 23 Jan 2020 21:14:37 +0100 Subject: [PATCH 15/36] change way studentinformation is gathered --- src/main.js | 26 +++++++++++++------------- 1 file changed, 13 insertions(+), 13 deletions(-) diff --git a/src/main.js b/src/main.js index 2234f20..41c3049 100644 --- a/src/main.js +++ b/src/main.js @@ -17,8 +17,8 @@ const CreditsByModuleTypeCount = { Zusatzmodul: { current: 0, min: 9 } }; -let studyTitle = "DEFAULT_TITLE" -let studyAcronym = "DEFAULT_ACRONYM" +let isPartTime = false +let studentInformations = {} // TODO: add more possible titles var StudyTitles = { @@ -54,17 +54,15 @@ function getStudyAcronym(studyTitle) { async function getStudyTitle() { let title = "If you see this message, something went wrong. Try to reload the page" - const URL = "https://mycampus.hslu.ch/de-ch/stud-i/mein-studium/meine-daten/" - - let data = await fetch((URL)) - .then(response => response.text()); +function getStudyTitle(studentData) { const searchStringStart = '

'; const searchStringEnd = "

"; - data = data.split(searchStringStart); - if (data[2]) { - title = data[2].split(searchStringEnd)[0].trim(); + studentData = studentData.split(searchStringStart); + + if (studentData[2]) { + title = studentData[2].split(searchStringEnd)[0].trim(); } return title; } @@ -207,7 +205,7 @@ function createTotalCreditsProgressBar(div) { function createStudyTitle(div) { let studyTitleTitle = document.createElement('h1'); - studyTitleTitle.appendChild(document.createTextNode('Studium: ' + studyTitle)); + studyTitleTitle.appendChild(document.createTextNode('Studium: ' + studentInformations.studyTitle)); div.insertBefore(studyTitleTitle, div.firstChild); } @@ -516,10 +514,12 @@ getStudyTitle().then(studyTitleText => { studyTitle = studyTitleText; studyAcronym = getStudyAcronym(studyTitle); - ModuleParser.generateModuleObjects(studyAcronym) +getStudentInformations() + .then( + ModuleParser.generateModuleObjects(studentInformations.studyAcronym) .then(modules => generateHtml(modules)) .catch(e => { console.log("Booo"); console.log(e); - }); -}); \ No newline at end of file + }) + ); \ No newline at end of file From 93426b7d733bee2c5014c8de05723c6ec6faf9f2 Mon Sep 17 00:00:00 2001 From: Letum Date: Thu, 23 Jan 2020 21:15:57 +0100 Subject: [PATCH 16/36] do some formating --- src/main.js | 137 +++++++++++++++++++++++++--------------------------- 1 file changed, 65 insertions(+), 72 deletions(-) diff --git a/src/main.js b/src/main.js index 41c3049..7757a19 100644 --- a/src/main.js +++ b/src/main.js @@ -7,7 +7,7 @@ const ModuleTableHeaders = [NameKey, ModuleTypeKey, CreditsKey, MarkKey, GradeKe const GradesCount = { A: 0, B: 0, C: 0, D: 0, E: 0, F: 0 }; //const CreditByTermCount = { 1: 0, 2: 0, 3: 0, 4: 0, 5: 0, 6: 0, 7: 0, 8: 0 }; -const CreditsBySemesterCount = [0,0,0,0,0,0,0,0]; +const CreditsBySemesterCount = [0, 0, 0, 0, 0, 0, 0, 0]; const CreditsByModuleTypeCount = { Kernmodul: { current: 0, min: 66 }, @@ -51,8 +51,6 @@ function getStudyAcronym(studyTitle) { * Gets the title of the students study * @returns {String} */ -async function getStudyTitle() { - let title = "If you see this message, something went wrong. Try to reload the page" function getStudyTitle(studentData) { @@ -269,11 +267,11 @@ function calculateStats(modules) { } const getBurndownValue = (semester) => { - if (CreditsBySemesterCount[semester-1] === 0) { + if (CreditsBySemesterCount[semester - 1] === 0) { // if no credits were achieved, don't show it in the graph return undefined; } - + let burndownValue = 180; for (let index = 0; index < semester; index++) { burndownValue -= CreditsBySemesterCount[index] @@ -283,7 +281,6 @@ const getBurndownValue = (semester) => { function createChart(div, modules) { canvas = document.createElement("canvas"); - canvas.setAttribute("id", "myChart"); div.insertBefore(canvas, div.firstChild); @@ -334,31 +331,31 @@ function createChart(div, modules) { }, { - label: 'Your remaining credits', - data: [ - getBurndownValue(0), - getBurndownValue(1), - getBurndownValue(2), - getBurndownValue(3), - getBurndownValue(4), - getBurndownValue(5), - getBurndownValue(6), - getBurndownValue(7), - getBurndownValue(8) - ], - backgroundColor: [ - 'rgba(135, 206, 235, 0.5)', - ], - borderColor: [ - 'rgba(135, 206, 235, 1)', - 'rgba(135, 206, 235, 1)', - 'rgba(135, 206, 235, 1)', - 'rgba(135, 206, 235, 1)', - 'rgba(135, 206, 235, 1)', - 'rgba(135, 206, 235, 1)', - 'rgba(135, 206, 235, 1)', - 'rgba(135, 206, 235, 1)' - ], + label: 'Your remaining credits', + data: [ + getBurndownValue(0), + getBurndownValue(1), + getBurndownValue(2), + getBurndownValue(3), + getBurndownValue(4), + getBurndownValue(5), + getBurndownValue(6), + getBurndownValue(7), + getBurndownValue(8) + ], + backgroundColor: [ + 'rgba(135, 206, 235, 0.5)', + ], + borderColor: [ + 'rgba(135, 206, 235, 1)', + 'rgba(135, 206, 235, 1)', + 'rgba(135, 206, 235, 1)', + 'rgba(135, 206, 235, 1)', + 'rgba(135, 206, 235, 1)', + 'rgba(135, 206, 235, 1)', + 'rgba(135, 206, 235, 1)', + 'rgba(135, 206, 235, 1)' + ], borderWidth: 2, yAxisID: 'y-axis-1' @@ -397,31 +394,31 @@ function createChart(div, modules) { } else { data.datasets.push( - { - label: 'Ideal remaining credits (full time)', - data: [ - 180, - 180 / 6 * 5, - 180 / 6 * 4, - 180 / 6 * 3, - 180 / 6 * 2, - 180 / 6 * 1, - 0 - ], - backgroundColor: [ - 'rgba(206, 255, 235, 0.5)', - ], - borderColor: [ - 'rgba(206, 255, 235, 1)', - 'rgba(206, 255, 235, 1)', - 'rgba(206, 255, 235, 1)', - 'rgba(206, 255, 235, 1)', - 'rgba(206, 255, 235, 1)', - 'rgba(206, 255, 235, 1)', - 'rgba(206, 255, 235, 1)', - 'rgba(206, 255, 235, 1)' - ], - borderWidth: 2 + { + label: 'Ideal remaining credits (full time)', + data: [ + 180, + 180 / 6 * 5, + 180 / 6 * 4, + 180 / 6 * 3, + 180 / 6 * 2, + 180 / 6 * 1, + 0 + ], + backgroundColor: [ + 'rgba(206, 255, 235, 0.5)', + ], + borderColor: [ + 'rgba(206, 255, 235, 1)', + 'rgba(206, 255, 235, 1)', + 'rgba(206, 255, 235, 1)', + 'rgba(206, 255, 235, 1)', + 'rgba(206, 255, 235, 1)', + 'rgba(206, 255, 235, 1)', + 'rgba(206, 255, 235, 1)', + 'rgba(206, 255, 235, 1)' + ], + borderWidth: 2 } ) } @@ -430,14 +427,14 @@ function createChart(div, modules) { scales: { yAxes: [ { - ticks: { - beginAtZero: true - }, - display: true, - position: 'left', - scaleLabel: { + ticks: { + beginAtZero: true + }, display: true, - labelString: 'Remaining credits' + position: 'left', + scaleLabel: { + display: true, + labelString: 'Remaining credits' }, id: 'y-axis-1', @@ -510,16 +507,12 @@ async function generateHtml(modules) { createStudyTitle(div); } -getStudyTitle().then(studyTitleText => { - studyTitle = studyTitleText; - studyAcronym = getStudyAcronym(studyTitle); - getStudentInformations() .then( ModuleParser.generateModuleObjects(studentInformations.studyAcronym) - .then(modules => generateHtml(modules)) - .catch(e => { - console.log("Booo"); - console.log(e); - }) + .then(modules => generateHtml(modules)) + .catch(e => { + console.log("Booo"); + console.log(e); + }) ); \ No newline at end of file From 9485db63c8461184555769ced9f460de749ff198 Mon Sep 17 00:00:00 2001 From: Letum Date: Thu, 23 Jan 2020 21:19:24 +0100 Subject: [PATCH 17/36] fix fulltime check --- src/main.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/main.js b/src/main.js index 7757a19..a9c3bf5 100644 --- a/src/main.js +++ b/src/main.js @@ -362,7 +362,7 @@ function createChart(div, modules) { } ] } - if (isPartTime) { + if (studentInformations.isPartTime) { data.datasets.push({ label: 'Ideal remaining credits (part time)', data: [ From c36e2106e86397297cbecc14309c91a40306b57c Mon Sep 17 00:00:00 2001 From: eddex Date: Wed, 29 Jan 2020 20:55:01 +0100 Subject: [PATCH 18/36] move license file --- src/LICENSE => LICENSE | 0 1 file changed, 0 insertions(+), 0 deletions(-) rename src/LICENSE => LICENSE (100%) diff --git a/src/LICENSE b/LICENSE similarity index 100% rename from src/LICENSE rename to LICENSE From 8017254d8c45997d88d5fee5b48a775ae01294a8 Mon Sep 17 00:00:00 2001 From: eddex Date: Wed, 29 Jan 2020 20:55:22 +0100 Subject: [PATCH 19/36] update --- LICENSE | 2 +- src/LICENSE | 6 ++++++ 2 files changed, 7 insertions(+), 1 deletion(-) create mode 100644 src/LICENSE diff --git a/LICENSE b/LICENSE index ddd6a0d..78aa392 100644 --- a/LICENSE +++ b/LICENSE @@ -1,6 +1,6 @@ MIT License -Copyright (c) 2019 Marco +Copyright (c) 2020 Marco & Brian Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal diff --git a/src/LICENSE b/src/LICENSE new file mode 100644 index 0000000..6917d4c --- /dev/null +++ b/src/LICENSE @@ -0,0 +1,6 @@ +Copyright (c) 2020 Marco & Brian + +Released under the MIT license. + +The full source code and license can be found at + https://github.com/eddex/hslu-simple-mep-results \ No newline at end of file From 169fcca0dfbf546eb40701bb9979bd2d0cd60a33 Mon Sep 17 00:00:00 2001 From: eddex Date: Thu, 30 Jan 2020 11:48:37 +0100 Subject: [PATCH 20/36] reorder credits title, chart and progress bar --- src/main.js | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/src/main.js b/src/main.js index a9c3bf5..816259d 100644 --- a/src/main.js +++ b/src/main.js @@ -497,13 +497,14 @@ async function generateHtml(modules) { createModulesTable(div, modules); Helpers.addTitleToDocument(div, 'Modulübersicht'); + await createGradesOverviewTable(div); + createAverageMarkTitle(div); + await createCreditsByModuleTypeTable(div); + createChart(div, modules); createTotalCreditsProgressBar(div); createTotalCreditsTitle(div); - await createGradesOverviewTable(div); - createAverageMarkTitle(div); - createChart(div, modules); createStudyTitle(div); } From 4d153df6874f4e79b7fab720c87579dd282dd62d Mon Sep 17 00:00:00 2001 From: eddex Date: Thu, 30 Jan 2020 13:05:16 +0100 Subject: [PATCH 21/36] refactor chart code and add comments :alien: --- src/main.js | 216 ++++++++++++++++------------------------------------ 1 file changed, 66 insertions(+), 150 deletions(-) diff --git a/src/main.js b/src/main.js index 816259d..ea4fdf1 100644 --- a/src/main.js +++ b/src/main.js @@ -6,8 +6,8 @@ const ModuleTypeKey = 'Modul-Typ'; const ModuleTableHeaders = [NameKey, ModuleTypeKey, CreditsKey, MarkKey, GradeKey] const GradesCount = { A: 0, B: 0, C: 0, D: 0, E: 0, F: 0 }; -//const CreditByTermCount = { 1: 0, 2: 0, 3: 0, 4: 0, 5: 0, 6: 0, 7: 0, 8: 0 }; -const CreditsBySemesterCount = [0, 0, 0, 0, 0, 0, 0, 0]; +// Semester 0 is the start. Will always be 0 credits. +const CreditsDoneBySemesterCount = [0, 0, 0, 0, 0, 0, 0, 0, 0]; const CreditsByModuleTypeCount = { Kernmodul: { current: 0, min: 66 }, @@ -267,163 +267,84 @@ function calculateStats(modules) { } const getBurndownValue = (semester) => { - if (CreditsBySemesterCount[semester - 1] === 0) { - // if no credits were achieved, don't show it in the graph - return undefined; - } - let burndownValue = 180; - for (let index = 0; index < semester; index++) { - burndownValue -= CreditsBySemesterCount[index] + for (let index = 0; index <= semester; index++) { + burndownValue -= CreditsDoneBySemesterCount[index] } return burndownValue; } +const getIdealBurndownDataForNumberOfSemesters = (numberOfSemesters) => { + let data = [] + for (let i = 0; i <= numberOfSemesters; i++) { + data.push(180 / numberOfSemesters * (numberOfSemesters-i)); + } + return data; +} + +/** + * Create a burndown chart that visualizes the remaining credits + * in comparison with the ideal remaining credits for each semester. + * On the same chart is a bar chart that shows how many credits were + * done each semester. + * + * @param div: The div to place the chart in. + * @param modules: A list of all modules of the student. + */ function createChart(div, modules) { - canvas = document.createElement("canvas"); + // the chart is drawn on this canvas + const canvas = document.createElement("canvas"); div.insertBefore(canvas, div.firstChild); - - const labels = [] - modules.forEach(modul => { - if (modul[GradeKey] != 'F' && modul.semester != undefined) { - CreditsBySemesterCount[modul.semester - 1] += Number(modul[CreditsKey]); + // calculate how many credits were achieved for each semester + modules.forEach(m => { + if (m[GradeKey] != 'F' && m.semester != undefined) { + CreditsDoneBySemesterCount[m.semester] += Number(m[CreditsKey]); } }) - for (let index = 0; index <= CreditsBySemesterCount.length; index++) { - labels.push(index) - } + // chart colors + const colorHsluDarkBlueTransparent = 'rgba(65, 94, 108, 0.5)'; + const colorLightBlueTransparent = 'rgba(135, 206, 235, 0.5)'; + const colorRedTransparent = 'rgba(255, 206, 235, 0.5)'; - let type = 'line' + // chart properties + const chartType = 'line' + const labels = CreditsDoneBySemesterCount.map((_, i, __) => 'Semester ' + i); + const yAxis2 = 'y-axis-2'; + const yAxis1 = 'y-axis-1'; - let data = { + let chartData = { labels: labels, datasets: [ { + // bars for credits achieved by semester label: 'Total Credits by Semster', - data: - [ - 0, - CreditsBySemesterCount[0], - CreditsBySemesterCount[1], - CreditsBySemesterCount[2], - CreditsBySemesterCount[3], - CreditsBySemesterCount[4], - CreditsBySemesterCount[5], - CreditsBySemesterCount[6], - CreditsBySemesterCount[7] - ], - - backgroundColor: [ - 'rgba(65, 94, 108, 0.8)', - 'rgba(65, 94, 108, 0.8)', - 'rgba(65, 94, 108, 0.8)', - 'rgba(65, 94, 108, 0.8)', - 'rgba(65, 94, 108, 0.8)', - 'rgba(65, 94, 108, 0.8)', - 'rgba(65, 94, 108, 0.8)', - 'rgba(65, 94, 108, 0.8)', - ], - // Changes this dataset to become a line - type: 'bar', - yAxisID: 'y-axis-2' - + data: CreditsDoneBySemesterCount, + backgroundColor: colorHsluDarkBlueTransparent, + type: 'bar', // only this dataset should be shown as bars + yAxisID: yAxis2 }, { + // burndown line for students remaining cretits label: 'Your remaining credits', - data: [ - getBurndownValue(0), - getBurndownValue(1), - getBurndownValue(2), - getBurndownValue(3), - getBurndownValue(4), - getBurndownValue(5), - getBurndownValue(6), - getBurndownValue(7), - getBurndownValue(8) - ], - backgroundColor: [ - 'rgba(135, 206, 235, 0.5)', - ], - borderColor: [ - 'rgba(135, 206, 235, 1)', - 'rgba(135, 206, 235, 1)', - 'rgba(135, 206, 235, 1)', - 'rgba(135, 206, 235, 1)', - 'rgba(135, 206, 235, 1)', - 'rgba(135, 206, 235, 1)', - 'rgba(135, 206, 235, 1)', - 'rgba(135, 206, 235, 1)' - ], - borderWidth: 2, - yAxisID: 'y-axis-1' - - } - ] - } - if (studentInformations.isPartTime) { - data.datasets.push({ - label: 'Ideal remaining credits (part time)', - data: [ - 180, - 180 / 8 * 7, - 180 / 8 * 6, - 180 / 8 * 5, - 180 / 8 * 4, - 180 / 8 * 3, - 180 / 8 * 2, - 180 / 8 * 1, - 0 - ], - backgroundColor: [ - 'rgba(255, 206, 235, 0.5)', - ], - borderColor: [ - 'rgba(255, 206, 235, 1)', - 'rgba(255, 206, 235, 1)', - 'rgba(255, 206, 235, 1)', - 'rgba(255, 206, 235, 1)', - 'rgba(255, 206, 235, 1)', - 'rgba(255, 206, 235, 1)', - 'rgba(255, 206, 235, 1)', - 'rgba(255, 206, 235, 1)' - ], - borderWidth: 2 - }) - } - else { - data.datasets.push( + data: CreditsDoneBySemesterCount.map((_, i, __) => getBurndownValue(i)), + backgroundColor: colorLightBlueTransparent, + yAxisID: yAxis1 + }, { - label: 'Ideal remaining credits (full time)', - data: [ - 180, - 180 / 6 * 5, - 180 / 6 * 4, - 180 / 6 * 3, - 180 / 6 * 2, - 180 / 6 * 1, - 0 - ], - backgroundColor: [ - 'rgba(206, 255, 235, 0.5)', - ], - borderColor: [ - 'rgba(206, 255, 235, 1)', - 'rgba(206, 255, 235, 1)', - 'rgba(206, 255, 235, 1)', - 'rgba(206, 255, 235, 1)', - 'rgba(206, 255, 235, 1)', - 'rgba(206, 255, 235, 1)', - 'rgba(206, 255, 235, 1)', - 'rgba(206, 255, 235, 1)' - ], - borderWidth: 2 + // reference burndown line to reach 0 remaining credits by the end of 6 or 8 semesters + label: 'Ideal remaining credits ' + (studentInformations.isPartTime ? '(part time)' : '(full time)'), + data: studentInformations.isPartTime + ? getIdealBurndownDataForNumberOfSemesters(8) + : getIdealBurndownDataForNumberOfSemesters(6), + backgroundColor: colorRedTransparent, + yAxisID: yAxis1 } - ) + ] } - let options = { + let chartOptions = { scales: { yAxes: [ { @@ -434,40 +355,35 @@ function createChart(div, modules) { position: 'left', scaleLabel: { display: true, - labelString: 'Remaining credits' + labelString: 'Total remaining credits' }, - id: 'y-axis-1', - + id: yAxis1 }, { ticks: { suggestedMin: 0, - suggestedMax: 180 + suggestedMax: 60 }, - type: 'linear', // only linear but allow scale type registration. This allows extensions to exist solely for log scale for instance + type: 'linear', display: true, position: 'right', scaleLabel: { display: true, - labelString: 'Credits by semster' + labelString: 'Credits done' }, - id: 'y-axis-2', + id: yAxis2, } ], xAxes: [{ - display: true, - scaleLabel: { - display: true, - labelString: 'Semester' - } + display: true }] } } new Chart(canvas, { - type: type, - data: data, - options: options + type: chartType, + data: chartData, + options: chartOptions }); } From 09e331381025656d798d0e724bec640a3c8ec975 Mon Sep 17 00:00:00 2001 From: eddex Date: Thu, 30 Jan 2020 14:00:03 +0100 Subject: [PATCH 22/36] refactor: unnecessarily global variable to local --- src/main.js | 18 +++++++++--------- 1 file changed, 9 insertions(+), 9 deletions(-) diff --git a/src/main.js b/src/main.js index ea4fdf1..f3eb068 100644 --- a/src/main.js +++ b/src/main.js @@ -6,9 +6,6 @@ const ModuleTypeKey = 'Modul-Typ'; const ModuleTableHeaders = [NameKey, ModuleTypeKey, CreditsKey, MarkKey, GradeKey] const GradesCount = { A: 0, B: 0, C: 0, D: 0, E: 0, F: 0 }; -// Semester 0 is the start. Will always be 0 credits. -const CreditsDoneBySemesterCount = [0, 0, 0, 0, 0, 0, 0, 0, 0]; - const CreditsByModuleTypeCount = { Kernmodul: { current: 0, min: 66 }, Projektmodul: { current: 0, min: 36 }, @@ -266,10 +263,10 @@ function calculateStats(modules) { }); } -const getBurndownValue = (semester) => { +const getBurndownValue = (creditsDoneBySemesterCount, semester) => { let burndownValue = 180; for (let index = 0; index <= semester; index++) { - burndownValue -= CreditsDoneBySemesterCount[index] + burndownValue -= creditsDoneBySemesterCount[index] } return burndownValue; } @@ -296,10 +293,13 @@ function createChart(div, modules) { const canvas = document.createElement("canvas"); div.insertBefore(canvas, div.firstChild); + // Semester 0 is the start. Will always be 0 credits. + const creditsDoneBySemesterCount = [0, 0, 0, 0, 0, 0, 0, 0, 0]; + // calculate how many credits were achieved for each semester modules.forEach(m => { if (m[GradeKey] != 'F' && m.semester != undefined) { - CreditsDoneBySemesterCount[m.semester] += Number(m[CreditsKey]); + creditsDoneBySemesterCount[m.semester] += Number(m[CreditsKey]); } }) @@ -310,7 +310,7 @@ function createChart(div, modules) { // chart properties const chartType = 'line' - const labels = CreditsDoneBySemesterCount.map((_, i, __) => 'Semester ' + i); + const labels = creditsDoneBySemesterCount.map((_, i, __) => 'Semester ' + i); const yAxis2 = 'y-axis-2'; const yAxis1 = 'y-axis-1'; @@ -320,7 +320,7 @@ function createChart(div, modules) { { // bars for credits achieved by semester label: 'Total Credits by Semster', - data: CreditsDoneBySemesterCount, + data: creditsDoneBySemesterCount, backgroundColor: colorHsluDarkBlueTransparent, type: 'bar', // only this dataset should be shown as bars yAxisID: yAxis2 @@ -328,7 +328,7 @@ function createChart(div, modules) { { // burndown line for students remaining cretits label: 'Your remaining credits', - data: CreditsDoneBySemesterCount.map((_, i, __) => getBurndownValue(i)), + data: creditsDoneBySemesterCount.map((_, i, __) => getBurndownValue(creditsDoneBySemesterCount, i)), backgroundColor: colorLightBlueTransparent, yAxisID: yAxis1 }, From 2593fa9ba6ec8a1ae3bc157be6eea58da1428603 Mon Sep 17 00:00:00 2001 From: eddex Date: Thu, 30 Jan 2020 17:36:58 +0100 Subject: [PATCH 23/36] keep scales the same to not give wrong info --- src/main.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/main.js b/src/main.js index f3eb068..7fd0d2c 100644 --- a/src/main.js +++ b/src/main.js @@ -362,7 +362,7 @@ function createChart(div, modules) { { ticks: { suggestedMin: 0, - suggestedMax: 60 + suggestedMax: 180 }, type: 'linear', display: true, From 9d64990c88b784ca0a5f769ea6a1fbbd62fe2400 Mon Sep 17 00:00:00 2001 From: eddex Date: Thu, 30 Jan 2020 17:57:05 +0100 Subject: [PATCH 24/36] more comments --- src/components/helpers.js | 21 +++++++++++++++++---- 1 file changed, 17 insertions(+), 4 deletions(-) diff --git a/src/components/helpers.js b/src/components/helpers.js index afb903a..bb6bceb 100644 --- a/src/components/helpers.js +++ b/src/components/helpers.js @@ -1,11 +1,15 @@ -/* +/** * Component for helper functions. */ const Helpers = { - /* + /** * Given a current value and a max value, calculate the percentage. * The result can be used for progress bars. + * + * @param current: The current value. + * @param max: The maximum value. + * @returns: A number between 0 and 100 representing the current progress. */ calculateProgress: (current, max) => { let progress = Math.round(current / max * 100); @@ -15,8 +19,12 @@ const Helpers = { return progress; }, - /* + /** * Add an h2 title as the first child of the given DOM element. + * + * @param element: The HTML element that should contain the title. + * Usually a div element. + * @param text: The text for the new title element. */ addTitleToDocument: (element, text) => { const title = document.createElement('h2'); @@ -24,10 +32,15 @@ const Helpers = { element.insertBefore(title, element.firstChild); }, - /* + /** * Helper method to read a file that is included in this browser extension. * The file needs to be registered in manifest.json! * Chome and Firefox have different APIs for this. + * + * @param filePath: The path to a file from the root of the extension (src/). + * e.g. data/file.json + * + * @returns: An URL that can be used to load the file with a web request. */ getExtensionInternalFileUrl: (filePath) => { From e95066818a819d190f7274b6f7c06d6f09eae0ab Mon Sep 17 00:00:00 2001 From: eddex Date: Thu, 30 Jan 2020 17:57:44 +0100 Subject: [PATCH 25/36] mark global variables with an underline for better readability --- src/components/module_parser.js | 24 +++---- src/main.js | 111 ++++++++++++++++---------------- 2 files changed, 67 insertions(+), 68 deletions(-) diff --git a/src/components/module_parser.js b/src/components/module_parser.js index d99c929..1b6f6d3 100644 --- a/src/components/module_parser.js +++ b/src/components/module_parser.js @@ -1,4 +1,4 @@ -/* +/** * Update the base module type list for other studies than information science. * * Sometimes the same module can have a different module type depending on the study (I, WI, ICS, DI etc). @@ -11,7 +11,7 @@ const updateModuleTypeList = async (oldModuleTypeList, jsonFilePath) => { const ModuleParser = { - /* + /** * Gets the value ("y") of a specified key ("x") in a 'detail' element of the API response. * detail: [ * key: "x", @@ -28,7 +28,7 @@ const ModuleParser = { return ''; }, - /* + /** * Check if a module was done in Autumn. * Modules are marked with 'H' for 'Herbstsemester' (autumn) * or 'F' for 'Frühlingssemester' (spring). @@ -39,7 +39,7 @@ const ModuleParser = { return includesH || includesF ? includesH : undefined; }, - /* + /** * Calculate the semester. */ calculateSemester: (hsluModule, firstModule) => { @@ -79,7 +79,7 @@ const ModuleParser = { return semester; }, - /* + /** * The module name includes the module id. * But there are different formats for the module names. * @@ -118,7 +118,7 @@ const ModuleParser = { } }, - /* + /** * Generates an array of module objects using the API and the module type mapping json file. */ generateModuleObjects: async (studyAcronym) => { @@ -155,19 +155,19 @@ const ModuleParser = { let passed = item.prop1[0].text == 'Erfolgreich teilgenommen'; parsedModule.passed = passed; - parsedModule[MarkKey] = item.note === null ? 'n/a' : item.note; - parsedModule[GradeKey] = item.grade === null ? 'n/a' : item.grade; - parsedModule[NameKey] = item.anlassnumber; + parsedModule[_MarkKey] = item.note === null ? 'n/a' : item.note; + parsedModule[_GradeKey] = item.grade === null ? 'n/a' : item.grade; + parsedModule[_NameKey] = item.anlassnumber; parsedModule.from = item.from; parsedModule.to = item.to; parsedModule.semester = ModuleParser.calculateSemester(item, firstModule); let details = item.details; - parsedModule[CreditsKey] = ModuleParser.getItemDetailsValueByKey(details, CreditsKey); + parsedModule[_CreditsKey] = ModuleParser.getItemDetailsValueByKey(details, _CreditsKey); - let moduleId = ModuleParser.getModuleIdFromModuleName(parsedModule[NameKey]); - parsedModule[ModuleTypeKey] = moduleTypeList[moduleId]; + let moduleId = ModuleParser.getModuleIdFromModuleName(parsedModule[_NameKey]); + parsedModule[_ModuleTypeKey] = moduleTypeList[moduleId]; modules.push(parsedModule); }); diff --git a/src/main.js b/src/main.js index 7fd0d2c..11c2c7f 100644 --- a/src/main.js +++ b/src/main.js @@ -1,12 +1,12 @@ -const NameKey = 'Modul-Name'; -const CreditsKey = 'ECTS-Punkte'; -const MarkKey = 'Bewertung'; -const GradeKey = 'Grad'; -const ModuleTypeKey = 'Modul-Typ'; -const ModuleTableHeaders = [NameKey, ModuleTypeKey, CreditsKey, MarkKey, GradeKey] -const GradesCount = { A: 0, B: 0, C: 0, D: 0, E: 0, F: 0 }; - -const CreditsByModuleTypeCount = { +const _NameKey = 'Modul-Name'; +const _CreditsKey = 'ECTS-Punkte'; +const _MarkKey = 'Bewertung'; +const _GradeKey = 'Grad'; +const _ModuleTypeKey = 'Modul-Typ'; +const _ModuleTableHeaders = [_NameKey, _ModuleTypeKey, _CreditsKey, _MarkKey, _GradeKey] +const _GradesCount = { A: 0, B: 0, C: 0, D: 0, E: 0, F: 0 }; + +const _CreditsByModuleTypeCount = { Kernmodul: { current: 0, min: 66 }, Projektmodul: { current: 0, min: 36 }, Erweiterungsmodul: { current: 0, min: 42 }, @@ -14,11 +14,10 @@ const CreditsByModuleTypeCount = { Zusatzmodul: { current: 0, min: 9 } }; -let isPartTime = false -let studentInformations = {} +const _StudentInformations = {} // TODO: add more possible titles -var StudyTitles = { +const _StudyTitles = { "bachelor of science in information & cyber security": "ICS", "bachelor of science in information": "I", "bachelor of science in computer science": "I", @@ -26,12 +25,12 @@ var StudyTitles = { "bachelor of science in informatik": "I" } -let totalCredits = 0; -let totalGrades = 0; -let totalNumericMark = 0; -let numberOfNumericMarks = 0; -let totalNumericMarkWithF = 0; -let numberOfNumericMarksWithF = 0; +let _totalCredits = 0; +let _totalGrades = 0; +let _totalNumericMark = 0; +let _numberOfNumericMarks = 0; +let _totalNumericMarkWithF = 0; +let _numberOfNumericMarksWithF = 0; /** * Gets the acronym of the students study. @@ -41,7 +40,7 @@ let numberOfNumericMarksWithF = 0; */ function getStudyAcronym(studyTitle) { studyTitle = studyTitle.toLowerCase().replace(/[0-9]/g, '').trim(); - return StudyTitles[studyTitle]; + return _StudyTitles[studyTitle]; } /** @@ -80,10 +79,10 @@ async function getStudentInformations() { let studentData = await fetch((URL)) .then(response => response.text()); - studentInformations.isPartTime = isPartTimeStudent(studentData); + _StudentInformations.isPartTime = isPartTimeStudent(studentData); - studentInformations.studyTitle = getStudyTitle(studentData); - studentInformations.studyAcronym = getStudyAcronym(studentInformations.studyTitle); + _StudentInformations.studyTitle = getStudyTitle(studentData); + _StudentInformations.studyAcronym = getStudyAcronym(_StudentInformations.studyTitle); } /* * Creates one row of the modules table. @@ -92,7 +91,7 @@ function createModulesTableRow(parsedModule) { let tr = document.createElement('tr'); - ModuleTableHeaders.forEach(attributeKey => { + _ModuleTableHeaders.forEach(attributeKey => { let td = document.createElement('td'); td.appendChild(document.createTextNode(parsedModule[attributeKey])); tr.appendChild(td); @@ -111,9 +110,9 @@ function createModulesTable(div, modules) { let header = table.createTHead(); let row = header.insertRow(0); - for (let i = 0; i < ModuleTableHeaders.length; i++) { + for (let i = 0; i < _ModuleTableHeaders.length; i++) { let cell = row.insertCell(i); - cell.appendChild(document.createTextNode(ModuleTableHeaders[i])); + cell.appendChild(document.createTextNode(_ModuleTableHeaders[i])); cell.setAttribute('style', 'font-weight: bold'); } @@ -139,15 +138,15 @@ async function createCreditsByModuleTypeTable(div) { creditsByModuleTypeTable.innerHTML = template; div.insertBefore(creditsByModuleTypeTable, div.firstChild); - for (let moduleKey in CreditsByModuleTypeCount) { + for (let moduleKey in _CreditsByModuleTypeCount) { const creditProgressBar = document.getElementById('ECTS-' + moduleKey); const creditProgressText = document.getElementById('ECTS-Text-' + moduleKey); const progress = Helpers.calculateProgress( - CreditsByModuleTypeCount[moduleKey].current, - CreditsByModuleTypeCount[moduleKey].min) + _CreditsByModuleTypeCount[moduleKey].current, + _CreditsByModuleTypeCount[moduleKey].min) creditProgressText.innerText = - CreditsByModuleTypeCount[moduleKey].current + ' (' + progress + '%)'; + _CreditsByModuleTypeCount[moduleKey].current + ' (' + progress + '%)'; creditProgressBar.style.width = progress + '%'; } } @@ -163,9 +162,9 @@ async function createGradesOverviewTable(div) { .then(response => response.text()); let gradeOverviewTable = document.createElement('div'); - for (let gradeId in GradesCount) { - gradesTableTemplate = String(gradesTableTemplate).replace('count-' + gradeId, GradesCount[gradeId]); - let gradePercentageRounded = Math.round(10000 * GradesCount[gradeId] / totalGrades) / 100; + for (let gradeId in _GradesCount) { + gradesTableTemplate = String(gradesTableTemplate).replace('count-' + gradeId, _GradesCount[gradeId]); + let gradePercentageRounded = Math.round(10000 * _GradesCount[gradeId] / _totalGrades) / 100; gradesTableTemplate = gradesTableTemplate.replace('percentage-' + gradeId, gradePercentageRounded + "%"); } gradeOverviewTable.innerHTML = gradesTableTemplate; @@ -176,8 +175,8 @@ async function createGradesOverviewTable(div) { * Create a heading that displays the number of achieved credits. */ function createTotalCreditsTitle(div) { - const progress = Helpers.calculateProgress(totalCredits, 180); - Helpers.addTitleToDocument(div, 'ECTS-Punkte: ' + totalCredits + '/180 (' + progress + '%)'); + const progress = Helpers.calculateProgress(_totalCredits, 180); + Helpers.addTitleToDocument(div, 'ECTS-Punkte: ' + _totalCredits + '/180 (' + progress + '%)'); } /* @@ -189,7 +188,7 @@ function createTotalCreditsProgressBar(div) { container.classList = 'total-progress-container'; const progressBar = document.createElement('div'); - const progress = Helpers.calculateProgress(totalCredits, 180); + const progress = Helpers.calculateProgress(_totalCredits, 180); progressBar.classList = 'total-progress progress'; progressBar.style.width = progress + '%'; @@ -200,7 +199,7 @@ function createTotalCreditsProgressBar(div) { function createStudyTitle(div) { let studyTitleTitle = document.createElement('h1'); - studyTitleTitle.appendChild(document.createTextNode('Studium: ' + studentInformations.studyTitle)); + studyTitleTitle.appendChild(document.createTextNode('Studium: ' + _StudentInformations.studyTitle)); div.insertBefore(studyTitleTitle, div.firstChild); } @@ -210,8 +209,8 @@ function createStudyTitle(div) { * A second average is displayed, where the modules with grade F are taken into account. */ function createAverageMarkTitle(div) { - let average = Number(totalNumericMark / numberOfNumericMarks).toFixed(2); - let averageWithF = Number(totalNumericMarkWithF / numberOfNumericMarksWithF).toFixed(2); + let average = Number(_totalNumericMark / _numberOfNumericMarks).toFixed(2); + let averageWithF = Number(_totalNumericMarkWithF / _numberOfNumericMarksWithF).toFixed(2); Helpers.addTitleToDocument(div, 'Noten Ø: ' + average + ' (Ø mit F: ' + averageWithF + ')') } @@ -238,26 +237,26 @@ function getExtensionInternalFileUrl(filePath) { function calculateStats(modules) { modules.forEach(parsedModule => { - if (parsedModule[GradeKey] != null && parsedModule[GradeKey] != '') { - GradesCount[parsedModule[GradeKey]]++; - totalGrades++; + if (parsedModule[_GradeKey] != null && parsedModule[_GradeKey] != '') { + _GradesCount[parsedModule[_GradeKey]]++; + _totalGrades++; } if (parsedModule.passed) { - let credits = Number(parsedModule[CreditsKey]); - totalCredits += credits; - let moduleType = parsedModule[ModuleTypeKey] - if (moduleType in CreditsByModuleTypeCount) { - CreditsByModuleTypeCount[moduleType].current += credits; + let credits = Number(parsedModule[_CreditsKey]); + _totalCredits += credits; + let moduleType = parsedModule[_ModuleTypeKey] + if (moduleType in _CreditsByModuleTypeCount) { + _CreditsByModuleTypeCount[moduleType].current += credits; } } // if cell is empty, Number('') returns 0! - numericMark = Number(parsedModule[MarkKey]) + numericMark = Number(parsedModule[_MarkKey]) if (!isNaN(numericMark) && numericMark != 0) { - totalNumericMarkWithF += numericMark; - numberOfNumericMarksWithF++; + _totalNumericMarkWithF += numericMark; + _numberOfNumericMarksWithF++; if (parsedModule.passed) { - totalNumericMark += numericMark; - numberOfNumericMarks++; + _totalNumericMark += numericMark; + _numberOfNumericMarks++; } } }); @@ -298,8 +297,8 @@ function createChart(div, modules) { // calculate how many credits were achieved for each semester modules.forEach(m => { - if (m[GradeKey] != 'F' && m.semester != undefined) { - creditsDoneBySemesterCount[m.semester] += Number(m[CreditsKey]); + if (m[_GradeKey] != 'F' && m.semester != undefined) { + creditsDoneBySemesterCount[m.semester] += Number(m[_CreditsKey]); } }) @@ -334,8 +333,8 @@ function createChart(div, modules) { }, { // reference burndown line to reach 0 remaining credits by the end of 6 or 8 semesters - label: 'Ideal remaining credits ' + (studentInformations.isPartTime ? '(part time)' : '(full time)'), - data: studentInformations.isPartTime + label: 'Ideal remaining credits ' + (_StudentInformations.isPartTime ? '(part time)' : '(full time)'), + data: _StudentInformations.isPartTime ? getIdealBurndownDataForNumberOfSemesters(8) : getIdealBurndownDataForNumberOfSemesters(6), backgroundColor: colorRedTransparent, @@ -426,7 +425,7 @@ async function generateHtml(modules) { getStudentInformations() .then( - ModuleParser.generateModuleObjects(studentInformations.studyAcronym) + ModuleParser.generateModuleObjects(_StudentInformations.studyAcronym) .then(modules => generateHtml(modules)) .catch(e => { console.log("Booo"); From 85a2331039c05660ec6e47b611154d9ca5fd70ce Mon Sep 17 00:00:00 2001 From: eddex Date: Thu, 30 Jan 2020 18:26:22 +0100 Subject: [PATCH 26/36] refactor: put global variables into single object --- src/main.js | 58 ++++++++++++++++++++++++++++------------------------- 1 file changed, 31 insertions(+), 27 deletions(-) diff --git a/src/main.js b/src/main.js index 11c2c7f..24fe3a7 100644 --- a/src/main.js +++ b/src/main.js @@ -14,7 +14,18 @@ const _CreditsByModuleTypeCount = { Zusatzmodul: { current: 0, min: 9 } }; -const _StudentInformations = {} +// Initialize student object with default values. +const _Student = { + isPartTime: false, + studyTitle: '', + studyAcronym: '', + totalCredits: 0, + totalGrades: 0, + totalNumericMark: 0, + numberOfNumericMarks: 0, + totalNumericMarkWithF: 0, + numberOfNumericMarksWithF: 0, +} // TODO: add more possible titles const _StudyTitles = { @@ -25,13 +36,6 @@ const _StudyTitles = { "bachelor of science in informatik": "I" } -let _totalCredits = 0; -let _totalGrades = 0; -let _totalNumericMark = 0; -let _numberOfNumericMarks = 0; -let _totalNumericMarkWithF = 0; -let _numberOfNumericMarksWithF = 0; - /** * Gets the acronym of the students study. * @param {String} studyTitle @@ -79,10 +83,10 @@ async function getStudentInformations() { let studentData = await fetch((URL)) .then(response => response.text()); - _StudentInformations.isPartTime = isPartTimeStudent(studentData); + _Student.isPartTime = isPartTimeStudent(studentData); - _StudentInformations.studyTitle = getStudyTitle(studentData); - _StudentInformations.studyAcronym = getStudyAcronym(_StudentInformations.studyTitle); + _Student.studyTitle = getStudyTitle(studentData); + _Student.studyAcronym = getStudyAcronym(_Student.studyTitle); } /* * Creates one row of the modules table. @@ -164,7 +168,7 @@ async function createGradesOverviewTable(div) { let gradeOverviewTable = document.createElement('div'); for (let gradeId in _GradesCount) { gradesTableTemplate = String(gradesTableTemplate).replace('count-' + gradeId, _GradesCount[gradeId]); - let gradePercentageRounded = Math.round(10000 * _GradesCount[gradeId] / _totalGrades) / 100; + let gradePercentageRounded = Math.round(10000 * _GradesCount[gradeId] / _Student.totalGrades) / 100; gradesTableTemplate = gradesTableTemplate.replace('percentage-' + gradeId, gradePercentageRounded + "%"); } gradeOverviewTable.innerHTML = gradesTableTemplate; @@ -175,8 +179,8 @@ async function createGradesOverviewTable(div) { * Create a heading that displays the number of achieved credits. */ function createTotalCreditsTitle(div) { - const progress = Helpers.calculateProgress(_totalCredits, 180); - Helpers.addTitleToDocument(div, 'ECTS-Punkte: ' + _totalCredits + '/180 (' + progress + '%)'); + const progress = Helpers.calculateProgress(_Student.totalCredits, 180); + Helpers.addTitleToDocument(div, 'ECTS-Punkte: ' + _Student.totalCredits + '/180 (' + progress + '%)'); } /* @@ -188,7 +192,7 @@ function createTotalCreditsProgressBar(div) { container.classList = 'total-progress-container'; const progressBar = document.createElement('div'); - const progress = Helpers.calculateProgress(_totalCredits, 180); + const progress = Helpers.calculateProgress(_Student.totalCredits, 180); progressBar.classList = 'total-progress progress'; progressBar.style.width = progress + '%'; @@ -199,7 +203,7 @@ function createTotalCreditsProgressBar(div) { function createStudyTitle(div) { let studyTitleTitle = document.createElement('h1'); - studyTitleTitle.appendChild(document.createTextNode('Studium: ' + _StudentInformations.studyTitle)); + studyTitleTitle.appendChild(document.createTextNode('Studium: ' + _Student.studyTitle)); div.insertBefore(studyTitleTitle, div.firstChild); } @@ -209,8 +213,8 @@ function createStudyTitle(div) { * A second average is displayed, where the modules with grade F are taken into account. */ function createAverageMarkTitle(div) { - let average = Number(_totalNumericMark / _numberOfNumericMarks).toFixed(2); - let averageWithF = Number(_totalNumericMarkWithF / _numberOfNumericMarksWithF).toFixed(2); + let average = Number(_Student.totalNumericMark / _Student.numberOfNumericMarks).toFixed(2); + let averageWithF = Number(_Student.totalNumericMarkWithF / _Student.numberOfNumericMarksWithF).toFixed(2); Helpers.addTitleToDocument(div, 'Noten Ø: ' + average + ' (Ø mit F: ' + averageWithF + ')') } @@ -239,11 +243,11 @@ function calculateStats(modules) { modules.forEach(parsedModule => { if (parsedModule[_GradeKey] != null && parsedModule[_GradeKey] != '') { _GradesCount[parsedModule[_GradeKey]]++; - _totalGrades++; + _Student.totalGrades++; } if (parsedModule.passed) { let credits = Number(parsedModule[_CreditsKey]); - _totalCredits += credits; + _Student.totalCredits += credits; let moduleType = parsedModule[_ModuleTypeKey] if (moduleType in _CreditsByModuleTypeCount) { _CreditsByModuleTypeCount[moduleType].current += credits; @@ -252,11 +256,11 @@ function calculateStats(modules) { // if cell is empty, Number('') returns 0! numericMark = Number(parsedModule[_MarkKey]) if (!isNaN(numericMark) && numericMark != 0) { - _totalNumericMarkWithF += numericMark; - _numberOfNumericMarksWithF++; + _Student.totalNumericMarkWithF += numericMark; + _Student.numberOfNumericMarksWithF++; if (parsedModule.passed) { - _totalNumericMark += numericMark; - _numberOfNumericMarks++; + _Student.totalNumericMark += numericMark; + _Student.numberOfNumericMarks++; } } }); @@ -333,8 +337,8 @@ function createChart(div, modules) { }, { // reference burndown line to reach 0 remaining credits by the end of 6 or 8 semesters - label: 'Ideal remaining credits ' + (_StudentInformations.isPartTime ? '(part time)' : '(full time)'), - data: _StudentInformations.isPartTime + label: 'Ideal remaining credits ' + (_Student.isPartTime ? '(part time)' : '(full time)'), + data: _Student.isPartTime ? getIdealBurndownDataForNumberOfSemesters(8) : getIdealBurndownDataForNumberOfSemesters(6), backgroundColor: colorRedTransparent, @@ -425,7 +429,7 @@ async function generateHtml(modules) { getStudentInformations() .then( - ModuleParser.generateModuleObjects(_StudentInformations.studyAcronym) + ModuleParser.generateModuleObjects(_Student.studyAcronym) .then(modules => generateHtml(modules)) .catch(e => { console.log("Booo"); From cf814b70ac31ce176aff697584640d79afd85aec Mon Sep 17 00:00:00 2001 From: eddex Date: Thu, 30 Jan 2020 18:41:34 +0100 Subject: [PATCH 27/36] refactor: remove global ModuleTableHeaders array --- src/main.js | 57 ++++++++++++++++++++++++++++++++++++----------------- 1 file changed, 39 insertions(+), 18 deletions(-) diff --git a/src/main.js b/src/main.js index 24fe3a7..8ebb82f 100644 --- a/src/main.js +++ b/src/main.js @@ -3,7 +3,6 @@ const _CreditsKey = 'ECTS-Punkte'; const _MarkKey = 'Bewertung'; const _GradeKey = 'Grad'; const _ModuleTypeKey = 'Modul-Typ'; -const _ModuleTableHeaders = [_NameKey, _ModuleTypeKey, _CreditsKey, _MarkKey, _GradeKey] const _GradesCount = { A: 0, B: 0, C: 0, D: 0, E: 0, F: 0 }; const _CreditsByModuleTypeCount = { @@ -88,37 +87,59 @@ async function getStudentInformations() { _Student.studyTitle = getStudyTitle(studentData); _Student.studyAcronym = getStudyAcronym(_Student.studyTitle); } -/* - * Creates one row of the modules table. + +/** + * Create one cell of the modules table. + * @param {string} text to put into the cell. */ -function createModulesTableRow(parsedModule) { +function createModulesTableCell(text) { + let td = document.createElement('td'); + td.appendChild(document.createTextNode(text)); + return td; +} +/** + * Create one row of the modules table. + */ +function createModulesTableRow(parsedModule) { let tr = document.createElement('tr'); - - _ModuleTableHeaders.forEach(attributeKey => { - let td = document.createElement('td'); - td.appendChild(document.createTextNode(parsedModule[attributeKey])); - tr.appendChild(td); - }); - + tr.appendChild(createModulesTableCell(parsedModule[_NameKey])); + tr.appendChild(createModulesTableCell(parsedModule[_ModuleTypeKey])); + tr.appendChild(createModulesTableCell(parsedModule[_CreditsKey])); + tr.appendChild(createModulesTableCell(parsedModule[_MarkKey])); + tr.appendChild(createModulesTableCell(parsedModule[_GradeKey])); return tr; } -/* +/** + * Insert a cell into the a row of a table. Text is bold. + * + * @param {number} index where to place the header cell. + * @param {any} row to insert cell. + * @param {string} text to write into the cell. + */ +function insertTableHeaderCell(index, row, text) { + let cell = row.insertCell(index); + cell.appendChild(document.createTextNode(text)); + cell.setAttribute('style', 'font-weight: bold'); +} + +/** * Dynamically creates a table that contains all modules the student has visited. * Shows Module Identifier (Nummer), Credits (ECTS), Module-Type, Numeric Mark (1-6) and Grade (A-F). + * */ function createModulesTable(div, modules) { let table = document.createElement('table'); let header = table.createTHead(); - let row = header.insertRow(0); - for (let i = 0; i < _ModuleTableHeaders.length; i++) { - let cell = row.insertCell(i); - cell.appendChild(document.createTextNode(_ModuleTableHeaders[i])); - cell.setAttribute('style', 'font-weight: bold'); - } + let headerRow = header.insertRow(0); + insertTableHeaderCell(0, headerRow, 'Modul-Name'); + insertTableHeaderCell(1, headerRow, 'Modul-Typ'); + insertTableHeaderCell(2, headerRow, 'ECTS-Punkte'); + insertTableHeaderCell(3, headerRow, 'Bewertung'); + insertTableHeaderCell(4, headerRow, 'Grad'); let tbody = document.createElement('tbody'); From 9ec29c409526337a0457bb7b6a068599820f4d7d Mon Sep 17 00:00:00 2001 From: eddex Date: Thu, 30 Jan 2020 18:52:57 +0100 Subject: [PATCH 28/36] refactor: replace global keys with attributes --- src/components/module_parser.js | 13 +++++++------ src/main.js | 29 ++++++++++++----------------- 2 files changed, 19 insertions(+), 23 deletions(-) diff --git a/src/components/module_parser.js b/src/components/module_parser.js index 1b6f6d3..403d5ae 100644 --- a/src/components/module_parser.js +++ b/src/components/module_parser.js @@ -155,19 +155,20 @@ const ModuleParser = { let passed = item.prop1[0].text == 'Erfolgreich teilgenommen'; parsedModule.passed = passed; - parsedModule[_MarkKey] = item.note === null ? 'n/a' : item.note; - parsedModule[_GradeKey] = item.grade === null ? 'n/a' : item.grade; - parsedModule[_NameKey] = item.anlassnumber; + parsedModule.mark = item.note === null ? 'n/a' : item.note; + parsedModule.grade = item.grade === null ? 'n/a' : item.grade; + parsedModule.name = item.anlassnumber; parsedModule.from = item.from; parsedModule.to = item.to; parsedModule.semester = ModuleParser.calculateSemester(item, firstModule); let details = item.details; - parsedModule[_CreditsKey] = ModuleParser.getItemDetailsValueByKey(details, _CreditsKey); + const creditsKey = 'ECTS-Punkte'; + parsedModule.credits = ModuleParser.getItemDetailsValueByKey(details, creditsKey); - let moduleId = ModuleParser.getModuleIdFromModuleName(parsedModule[_NameKey]); - parsedModule[_ModuleTypeKey] = moduleTypeList[moduleId]; + let moduleId = ModuleParser.getModuleIdFromModuleName(parsedModule.name); + parsedModule.moduleType = moduleTypeList[moduleId]; modules.push(parsedModule); }); diff --git a/src/main.js b/src/main.js index 8ebb82f..f3e34e8 100644 --- a/src/main.js +++ b/src/main.js @@ -1,8 +1,3 @@ -const _NameKey = 'Modul-Name'; -const _CreditsKey = 'ECTS-Punkte'; -const _MarkKey = 'Bewertung'; -const _GradeKey = 'Grad'; -const _ModuleTypeKey = 'Modul-Typ'; const _GradesCount = { A: 0, B: 0, C: 0, D: 0, E: 0, F: 0 }; const _CreditsByModuleTypeCount = { @@ -103,11 +98,11 @@ function createModulesTableCell(text) { */ function createModulesTableRow(parsedModule) { let tr = document.createElement('tr'); - tr.appendChild(createModulesTableCell(parsedModule[_NameKey])); - tr.appendChild(createModulesTableCell(parsedModule[_ModuleTypeKey])); - tr.appendChild(createModulesTableCell(parsedModule[_CreditsKey])); - tr.appendChild(createModulesTableCell(parsedModule[_MarkKey])); - tr.appendChild(createModulesTableCell(parsedModule[_GradeKey])); + tr.appendChild(createModulesTableCell(parsedModule.name)); + tr.appendChild(createModulesTableCell(parsedModule.moduleType)); + tr.appendChild(createModulesTableCell(parsedModule.credits)); + tr.appendChild(createModulesTableCell(parsedModule.mark)); + tr.appendChild(createModulesTableCell(parsedModule.grade)); return tr; } @@ -262,20 +257,20 @@ function getExtensionInternalFileUrl(filePath) { function calculateStats(modules) { modules.forEach(parsedModule => { - if (parsedModule[_GradeKey] != null && parsedModule[_GradeKey] != '') { - _GradesCount[parsedModule[_GradeKey]]++; + if (parsedModule.grade != null && parsedModule.grade != '') { + _GradesCount[parsedModule.garde]++; _Student.totalGrades++; } if (parsedModule.passed) { - let credits = Number(parsedModule[_CreditsKey]); + let credits = Number(parsedModule.credits); _Student.totalCredits += credits; - let moduleType = parsedModule[_ModuleTypeKey] + let moduleType = parsedModule.moduleType; if (moduleType in _CreditsByModuleTypeCount) { _CreditsByModuleTypeCount[moduleType].current += credits; } } // if cell is empty, Number('') returns 0! - numericMark = Number(parsedModule[_MarkKey]) + numericMark = Number(parsedModule.mark) if (!isNaN(numericMark) && numericMark != 0) { _Student.totalNumericMarkWithF += numericMark; _Student.numberOfNumericMarksWithF++; @@ -322,8 +317,8 @@ function createChart(div, modules) { // calculate how many credits were achieved for each semester modules.forEach(m => { - if (m[_GradeKey] != 'F' && m.semester != undefined) { - creditsDoneBySemesterCount[m.semester] += Number(m[_CreditsKey]); + if (m.grade != 'F' && m.semester != undefined) { + creditsDoneBySemesterCount[m.semester] += Number(m.credits); } }) From 59d10b333596f3452aad04c880d23c4aadc10930 Mon Sep 17 00:00:00 2001 From: eddex Date: Thu, 30 Jan 2020 19:21:17 +0100 Subject: [PATCH 29/36] refactor: move gradesCount into _Student object --- src/main.js | 17 ++++++++++------- 1 file changed, 10 insertions(+), 7 deletions(-) diff --git a/src/main.js b/src/main.js index f3e34e8..7d50cba 100644 --- a/src/main.js +++ b/src/main.js @@ -1,5 +1,3 @@ -const _GradesCount = { A: 0, B: 0, C: 0, D: 0, E: 0, F: 0 }; - const _CreditsByModuleTypeCount = { Kernmodul: { current: 0, min: 66 }, Projektmodul: { current: 0, min: 36 }, @@ -19,6 +17,7 @@ const _Student = { numberOfNumericMarks: 0, totalNumericMarkWithF: 0, numberOfNumericMarksWithF: 0, + gradesCount: { A: 0, B: 0, C: 0, D: 0, E: 0, F: 0 } } // TODO: add more possible titles @@ -182,9 +181,9 @@ async function createGradesOverviewTable(div) { .then(response => response.text()); let gradeOverviewTable = document.createElement('div'); - for (let gradeId in _GradesCount) { - gradesTableTemplate = String(gradesTableTemplate).replace('count-' + gradeId, _GradesCount[gradeId]); - let gradePercentageRounded = Math.round(10000 * _GradesCount[gradeId] / _Student.totalGrades) / 100; + for (let gradeId in _Student.gradesCount) { + gradesTableTemplate = String(gradesTableTemplate).replace('count-' + gradeId, _Student.gradesCount[gradeId]); + let gradePercentageRounded = Math.round(10000 * _Student.gradesCount[gradeId] / _Student.totalGrades) / 100; gradesTableTemplate = gradesTableTemplate.replace('percentage-' + gradeId, gradePercentageRounded + "%"); } gradeOverviewTable.innerHTML = gradesTableTemplate; @@ -257,8 +256,12 @@ function getExtensionInternalFileUrl(filePath) { function calculateStats(modules) { modules.forEach(parsedModule => { - if (parsedModule.grade != null && parsedModule.grade != '') { - _GradesCount[parsedModule.garde]++; + if (parsedModule.grade + && parsedModule.grade != null + && parsedModule.grade != '' + && parsedModule.grade != 'n/a') { + console.log(parsedModule.grade) + _Student.gradesCount[parsedModule.grade]++; _Student.totalGrades++; } if (parsedModule.passed) { From 6e73cf4cb21b44ea613c95ea02fb64b553e17032 Mon Sep 17 00:00:00 2001 From: eddex Date: Thu, 30 Jan 2020 19:29:44 +0100 Subject: [PATCH 30/36] refactor: put creditsByModuleTypeCount into _Student --- src/main.js | 49 ++++++++++++++++++++++--------------------------- 1 file changed, 22 insertions(+), 27 deletions(-) diff --git a/src/main.js b/src/main.js index 7d50cba..be437af 100644 --- a/src/main.js +++ b/src/main.js @@ -1,11 +1,3 @@ -const _CreditsByModuleTypeCount = { - Kernmodul: { current: 0, min: 66 }, - Projektmodul: { current: 0, min: 36 }, - Erweiterungsmodul: { current: 0, min: 42 }, - Majormodul: { current: 0, min: 24 }, - Zusatzmodul: { current: 0, min: 9 } -}; - // Initialize student object with default values. const _Student = { isPartTime: false, @@ -17,27 +9,31 @@ const _Student = { numberOfNumericMarks: 0, totalNumericMarkWithF: 0, numberOfNumericMarksWithF: 0, - gradesCount: { A: 0, B: 0, C: 0, D: 0, E: 0, F: 0 } -} - -// TODO: add more possible titles -const _StudyTitles = { - "bachelor of science in information & cyber security": "ICS", - "bachelor of science in information": "I", - "bachelor of science in computer science": "I", - "bachelor of science in wirtschaftsinformatik": "WI", - "bachelor of science in informatik": "I" + gradesCount: { A: 0, B: 0, C: 0, D: 0, E: 0, F: 0 }, + creditsByModuleTypeCount: { + Kernmodul: { current: 0, min: 66 }, + Projektmodul: { current: 0, min: 36 }, + Erweiterungsmodul: { current: 0, min: 42 }, + Majormodul: { current: 0, min: 24 }, + Zusatzmodul: { current: 0, min: 9 } + } } /** * Gets the acronym of the students study. * @param {String} studyTitle * @returns {String} I, ICS or WI - * */ function getStudyAcronym(studyTitle) { + const studyTitles = { + "bachelor of science in information & cyber security": "ICS", + "bachelor of science in information": "I", + "bachelor of science in computer science": "I", + "bachelor of science in wirtschaftsinformatik": "WI", + "bachelor of science in informatik": "I" + } studyTitle = studyTitle.toLowerCase().replace(/[0-9]/g, '').trim(); - return _StudyTitles[studyTitle]; + return studyTitles[studyTitle]; } /** @@ -157,15 +153,15 @@ async function createCreditsByModuleTypeTable(div) { creditsByModuleTypeTable.innerHTML = template; div.insertBefore(creditsByModuleTypeTable, div.firstChild); - for (let moduleKey in _CreditsByModuleTypeCount) { + for (let moduleKey in _Student.creditsByModuleTypeCount) { const creditProgressBar = document.getElementById('ECTS-' + moduleKey); const creditProgressText = document.getElementById('ECTS-Text-' + moduleKey); const progress = Helpers.calculateProgress( - _CreditsByModuleTypeCount[moduleKey].current, - _CreditsByModuleTypeCount[moduleKey].min) + _Student.creditsByModuleTypeCount[moduleKey].current, + _Student.creditsByModuleTypeCount[moduleKey].min) creditProgressText.innerText = - _CreditsByModuleTypeCount[moduleKey].current + ' (' + progress + '%)'; + _Student.creditsByModuleTypeCount[moduleKey].current + ' (' + progress + '%)'; creditProgressBar.style.width = progress + '%'; } } @@ -260,7 +256,6 @@ function calculateStats(modules) { && parsedModule.grade != null && parsedModule.grade != '' && parsedModule.grade != 'n/a') { - console.log(parsedModule.grade) _Student.gradesCount[parsedModule.grade]++; _Student.totalGrades++; } @@ -268,8 +263,8 @@ function calculateStats(modules) { let credits = Number(parsedModule.credits); _Student.totalCredits += credits; let moduleType = parsedModule.moduleType; - if (moduleType in _CreditsByModuleTypeCount) { - _CreditsByModuleTypeCount[moduleType].current += credits; + if (moduleType in _Student.creditsByModuleTypeCount) { + _Student.creditsByModuleTypeCount[moduleType].current += credits; } } // if cell is empty, Number('') returns 0! From 6d446971a1292fed8147a60d48fb4aa0f2ee9e53 Mon Sep 17 00:00:00 2001 From: eddex Date: Thu, 30 Jan 2020 19:38:42 +0100 Subject: [PATCH 31/36] =?UTF-8?q?feat:=20add=20'Modultypen=20=C3=9Cbersich?= =?UTF-8?q?t'=20title?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/main.js | 1 + 1 file changed, 1 insertion(+) diff --git a/src/main.js b/src/main.js index be437af..6d250cf 100644 --- a/src/main.js +++ b/src/main.js @@ -434,6 +434,7 @@ async function generateHtml(modules) { createAverageMarkTitle(div); await createCreditsByModuleTypeTable(div); + Helpers.addTitleToDocument(div, 'Modultypen Übersicht'); createChart(div, modules); createTotalCreditsProgressBar(div); createTotalCreditsTitle(div); From 88171c810ceeb17b427692b36703d870302d8847 Mon Sep 17 00:00:00 2001 From: eddex Date: Thu, 30 Jan 2020 21:02:12 +0100 Subject: [PATCH 32/36] refactor: change orientation of module type grid and remove progress bars --- src/main.js | 33 ++++++++++++------- .../credits_by_module_type_table.html | 29 ++++------------ src/templates/custom_styles.css | 21 +++++------- 3 files changed, 37 insertions(+), 46 deletions(-) diff --git a/src/main.js b/src/main.js index 6d250cf..8eea6b4 100644 --- a/src/main.js +++ b/src/main.js @@ -142,7 +142,22 @@ function createModulesTable(div, modules) { div.insertBefore(table, div.firstChild); } -/* +/** + * Add the progress of the credits for a module type to the template. + * The template must already be present on the page! + * + * @param {string} elementId: The id of the html element to add the text to. + * @param {any} creditsByModuleType: An object containing the current and min credits of a module type. + */ +function addProgressTextToTemplate(elementId, creditsByModuleType) { + const modultypeProgressDiv = document.getElementById(elementId); + const progress = Helpers.calculateProgress( + creditsByModuleType.current, + creditsByModuleType.min); + modultypeProgressDiv.innerText = creditsByModuleType.current + ' (' + progress + '%)'; +} + +/** * Create a table that shows how many ECTS for each type of module have been achieved. */ async function createCreditsByModuleTypeTable(div) { @@ -153,17 +168,11 @@ async function createCreditsByModuleTypeTable(div) { creditsByModuleTypeTable.innerHTML = template; div.insertBefore(creditsByModuleTypeTable, div.firstChild); - for (let moduleKey in _Student.creditsByModuleTypeCount) { - const creditProgressBar = document.getElementById('ECTS-' + moduleKey); - const creditProgressText = document.getElementById('ECTS-Text-' + moduleKey); - - const progress = Helpers.calculateProgress( - _Student.creditsByModuleTypeCount[moduleKey].current, - _Student.creditsByModuleTypeCount[moduleKey].min) - creditProgressText.innerText = - _Student.creditsByModuleTypeCount[moduleKey].current + ' (' + progress + '%)'; - creditProgressBar.style.width = progress + '%'; - } + addProgressTextToTemplate('ECTS-Text-Kernmodul', _Student.creditsByModuleTypeCount.Kernmodul); + addProgressTextToTemplate('ECTS-Text-Erweiterungsmodul', _Student.creditsByModuleTypeCount.Erweiterungsmodul); + addProgressTextToTemplate('ECTS-Text-Majormodul', _Student.creditsByModuleTypeCount.Majormodul); + addProgressTextToTemplate('ECTS-Text-Projektmodul', _Student.creditsByModuleTypeCount.Projektmodul); + addProgressTextToTemplate('ECTS-Text-Zusatzmodul', _Student.creditsByModuleTypeCount.Zusatzmodul); } /* diff --git a/src/templates/credits_by_module_type_table.html b/src/templates/credits_by_module_type_table.html index 811a290..cdf76a6 100644 --- a/src/templates/credits_by_module_type_table.html +++ b/src/templates/credits_by_module_type_table.html @@ -1,28 +1,13 @@
-
Kernmodule (min 66)
-
- - -
+
Kernmodule (min 66)
Majormodule* (min 24)
-
- - -
Erweiterungsmodule (min 42)
-
- - -
Projektmodule (min 36)
-
- - -
Zusatzmodule (min 9)
-
- - -
+
+
+
+
+
-

*Majormodule zählen als Erweiterungsmodule. Falls alle 24 ECTS eines Majors erreicht wurden, erhält man diesen.

+

*Majormodule sind spezielle Erweiterungsmodule. Falls alle 24 ECTS eines Majors erreicht wurden, erhält man diesen.

diff --git a/src/templates/custom_styles.css b/src/templates/custom_styles.css index 667c243..ffc0b7d 100644 --- a/src/templates/custom_styles.css +++ b/src/templates/custom_styles.css @@ -1,3 +1,4 @@ +/* Modules table & grades table */ table, th, td { border: 1px solid #415e6c; } @@ -10,18 +11,12 @@ table { td { padding-left: 2px; } +/* END */ -.td-progress { - padding-left: 0px; -} - -.no-margin { - margin-bottom: 0rem; -} - +/* Credits by module type grid layout */ .progress-container { display: grid; - grid-template-columns: fit-content(100%) 5fr; + grid-template-columns: auto auto auto auto auto; grid-gap: 1px; padding: 1px; background-color: #415e6c; @@ -38,14 +33,15 @@ td { } .progress-text { - position: absolute; - margin-left: 2px; + padding-left: 2px; } .progress-title { padding-left: 2px; } +/* END */ +/* Total credits progress bar */ .total-progress-container { height: 10px; width: 100%; @@ -56,4 +52,5 @@ td { .total-progress { background-color: skyblue; height: 100%; -} \ No newline at end of file +} +/* END */ \ No newline at end of file From 60a887c2cb9875147ee846ba315a7a010e708e93 Mon Sep 17 00:00:00 2001 From: eddex Date: Thu, 30 Jan 2020 21:19:22 +0100 Subject: [PATCH 33/36] feat: padding around everything + smaller chart --- src/main.js | 21 +++++++++++++-------- src/templates/custom_styles.css | 14 ++++++++++++++ 2 files changed, 27 insertions(+), 8 deletions(-) diff --git a/src/main.js b/src/main.js index 8eea6b4..ac7e8f8 100644 --- a/src/main.js +++ b/src/main.js @@ -316,8 +316,11 @@ const getIdealBurndownDataForNumberOfSemesters = (numberOfSemesters) => { */ function createChart(div, modules) { // the chart is drawn on this canvas - const canvas = document.createElement("canvas"); - div.insertBefore(canvas, div.firstChild); + const canvas = document.createElement('canvas'); + const canvasDiv = document.createElement('div'); + canvasDiv.classList.add('chart-canvas'); + canvasDiv.appendChild(canvas); + div.insertBefore(canvasDiv, div.firstChild); // Semester 0 is the start. Will always be 0 credits. const creditsDoneBySemesterCount = [0, 0, 0, 0, 0, 0, 0, 0, 0]; @@ -420,17 +423,19 @@ async function generateHtml(modules) { document.getElementById('intro').remove(); document.getElementsByClassName('sidebar medium-7 columns mobile-column')[0].remove(); } catch (error) { - console.log("NOTITLE, DONT CARE") + console.log("NOTITLE, DONT CARE"); } let div = document.getElementsByClassName('row teaser-section None')[0]; + div.classList.add('allmighty-container'); + if (!modules) { - // API call was blocked. + // API call was blocked. This has been fixed. See issue #16. const p = document.createElement('p'); - p.innerHTML = 'HSLU Simple MEP Results Extension failed to load. The API \ - has blocked the request. Please try again later. \ - More infos on GitHub: Issue #16'; + p.innerHTML = 'HSLU Simple MEP Results Extension failed to load. \ + Consider to report the issue on GitHub if this happens regularly: \ + Create a new issue..'; div.insertBefore(p, div.firstChild); return; } diff --git a/src/templates/custom_styles.css b/src/templates/custom_styles.css index ffc0b7d..49d679d 100644 --- a/src/templates/custom_styles.css +++ b/src/templates/custom_styles.css @@ -1,3 +1,17 @@ +/* Container */ +.allmighty-container { + padding-left: 5px; + padding-right: 5px; +} +/* END */ + +/* Chart */ +.chart-canvas { + width: 90%; + margin: auto; + margin-bottom: 1.5rem; +} + /* Modules table & grades table */ table, th, td { border: 1px solid #415e6c; From 4461748a66ba2347f14f80dd0679208817a8938c Mon Sep 17 00:00:00 2001 From: eddex Date: Thu, 30 Jan 2020 21:44:33 +0100 Subject: [PATCH 34/36] feat: style tables more HSLU like --- src/main.js | 18 +++++++++--------- src/templates/custom_styles.css | 14 ++++++++++---- src/templates/grades_table.html | 3 +++ 3 files changed, 22 insertions(+), 13 deletions(-) diff --git a/src/main.js b/src/main.js index ac7e8f8..111f282 100644 --- a/src/main.js +++ b/src/main.js @@ -108,10 +108,10 @@ function createModulesTableRow(parsedModule) { * @param {any} row to insert cell. * @param {string} text to write into the cell. */ -function insertTableHeaderCell(index, row, text) { - let cell = row.insertCell(index); - cell.appendChild(document.createTextNode(text)); - cell.setAttribute('style', 'font-weight: bold'); +function insertTableHeaderCell(row, text) { + const th = document.createElement('th'); + th.innerText = text; + row.appendChild(th); } /** @@ -125,11 +125,11 @@ function createModulesTable(div, modules) { let header = table.createTHead(); let headerRow = header.insertRow(0); - insertTableHeaderCell(0, headerRow, 'Modul-Name'); - insertTableHeaderCell(1, headerRow, 'Modul-Typ'); - insertTableHeaderCell(2, headerRow, 'ECTS-Punkte'); - insertTableHeaderCell(3, headerRow, 'Bewertung'); - insertTableHeaderCell(4, headerRow, 'Grad'); + insertTableHeaderCell(headerRow, 'Modul-Name'); + insertTableHeaderCell(headerRow, 'Modul-Typ'); + insertTableHeaderCell(headerRow, 'ECTS-Punkte'); + insertTableHeaderCell(headerRow, 'Bewertung'); + insertTableHeaderCell(headerRow, 'Grad'); let tbody = document.createElement('tbody'); diff --git a/src/templates/custom_styles.css b/src/templates/custom_styles.css index 49d679d..df2b183 100644 --- a/src/templates/custom_styles.css +++ b/src/templates/custom_styles.css @@ -17,6 +17,12 @@ table, th, td { border: 1px solid #415e6c; } +th { + background-color: #415e6c; + color: white; + font-weight: normal; +} + table { margin-bottom: 1.6rem; width: 100%; @@ -36,10 +42,6 @@ td { background-color: #415e6c; } -.progress-container>div { - background-color: white; -} - .progress-bar { background-color: skyblue; display: block; @@ -48,10 +50,14 @@ td { .progress-text { padding-left: 2px; + background-color: white; } .progress-title { padding-left: 2px; + /* font-weight: bold; */ + color: white; + background-color: #415e6c; } /* END */ diff --git a/src/templates/grades_table.html b/src/templates/grades_table.html index d3801ca..1947f50 100644 --- a/src/templates/grades_table.html +++ b/src/templates/grades_table.html @@ -1,6 +1,7 @@ + @@ -9,6 +10,7 @@ + @@ -17,6 +19,7 @@ + From f2237b79f4f7b7c7d078d298f006a63fbbfa1ac2 Mon Sep 17 00:00:00 2001 From: eddex Date: Thu, 30 Jan 2020 21:56:18 +0100 Subject: [PATCH 35/36] feat: alternating colors of table rows --- src/main.js | 5 ++++- src/templates/custom_styles.css | 4 ++++ src/templates/grades_table.html | 6 +++--- 3 files changed, 11 insertions(+), 4 deletions(-) diff --git a/src/main.js b/src/main.js index 111f282..deb1161 100644 --- a/src/main.js +++ b/src/main.js @@ -133,8 +133,11 @@ function createModulesTable(div, modules) { let tbody = document.createElement('tbody'); - modules.forEach(parsedModule => { + modules.map((parsedModule, index) => { let tr = createModulesTableRow(parsedModule); + if (index % 2 == 0) { + tr.classList.add('colored-row'); + } tbody.appendChild(tr); }); diff --git a/src/templates/custom_styles.css b/src/templates/custom_styles.css index df2b183..db1188f 100644 --- a/src/templates/custom_styles.css +++ b/src/templates/custom_styles.css @@ -31,6 +31,10 @@ table { td { padding-left: 2px; } + +.colored-row { + background-color: #dfe2e5; +} /* END */ /* Credits by module type grid layout */ diff --git a/src/templates/grades_table.html b/src/templates/grades_table.html index 1947f50..4a9c2b7 100644 --- a/src/templates/grades_table.html +++ b/src/templates/grades_table.html @@ -9,8 +9,8 @@ - - + + @@ -19,7 +19,7 @@ - + From 954e02c88d35575d4a6ff11150f721a9fab6823e Mon Sep 17 00:00:00 2001 From: eddex Date: Thu, 30 Jan 2020 22:01:33 +0100 Subject: [PATCH 36/36] feat: fat red error message!!!! --- src/main.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/main.js b/src/main.js index deb1161..ab8a6a0 100644 --- a/src/main.js +++ b/src/main.js @@ -1,7 +1,7 @@ // Initialize student object with default values. const _Student = { isPartTime: false, - studyTitle: '', + studyTitle: 'Fehler! Bitte die Seite neu laden..', studyAcronym: '', totalCredits: 0, totalGrades: 0, @@ -226,7 +226,7 @@ function createTotalCreditsProgressBar(div) { function createStudyTitle(div) { let studyTitleTitle = document.createElement('h1'); - studyTitleTitle.appendChild(document.createTextNode('Studium: ' + _Student.studyTitle)); + studyTitleTitle.innerHTML = 'Studium: ' + _Student.studyTitle; div.insertBefore(studyTitleTitle, div.firstChild); }
A B CF
Count count-A count-B count-Ccount-F
Distribution percentage-A percentage-B percentage-CE F
Count
Anzahl count-A count-B count-Ccount-F
DistributionVerteilung percentage-A percentage-B percentage-C