';
+ if (!proc.active) phtml = '
';
+ phtml += "
Process " + i + ":
";
+ phtml += "
PID: " + (proc.pid||"None (not active)") + "
";
+ if (proc.threads && proc.active) {
+ phtml += "
Thread ID | Access count | Bytes served | Last Used | Last client | Last request |
";
+ for (var j in proc.threads) {
+ var thread = proc.threads[j];
+ thread.request = (thread.request||"(Unknown)").replace(/[<>]+/g, "");
+ phtml += ""+thread.thread+" | "+thread.count+" | "+thread.bytes+" | "+thread.last_used+" | "+thread.client+" | "+thread.request+" |
";
+ }
+ phtml += "
";
+ } else {
+ phtml += "
No thread information avaialable
";
+ }
+ phtml += "
";
+ box.innerHTML += phtml;
+ }
+}
+
+function waitTwo() {
+ getAsync(location.href + "?view=json&rnd=" + Math.random(), null, refreshCharts)
+}
+
+ function showPanel(what) {
+ var items = ['dashboard','misc','threads','modules'];
+ for (var i in items) {
+ var item = items[i];
+ var btn = document.getElementById(item+'_button');
+ var panel = document.getElementById(item+'_panel');
+ if (item == what) {
+ btn.setAttribute("class", "btn active");
+ panel.style.display = 'block';
+ } else {
+ btn.setAttribute("class", "btn");
+ panel.style.display = 'none';
+ }
+ }
+
+ // special constructors
+ if (what == 'threads') {
+ getAsync(location.href + "?view=json&extended=true&rnd=" + Math.random(), null, refreshThreads)
+ }
+ }
+
+ function fn(num) {
+ num = num + "";
+ num = num.replace(/(\d)(\d{9})$/, '$1,$2');
+ num = num.replace(/(\d)(\d{6})$/, '$1,$2');
+ num = num.replace(/(\d)(\d{3})$/, '$1,$2');
+ return num;
+ }
+
+ function fnmb(num) {
+ var add = "bytes";
+ var dec = "";
+ var mul = 1;
+ if (num > 1024) { add = "KB"; mul= 1024; }
+ if (num > (1024*1024)) { add = "MB"; mul= 1024*1024; }
+ if (num > (1024*1024*1024)) { add = "GB"; mul= 1024*1024*1024; }
+ if (num > (1024*1024*1024*1024)) { add = "TB"; mul= 1024*1024*1024*1024; }
+ num = num / mul;
+ if (add != "bytes") {
+ dec = "." + Math.floor( (num - Math.floor(num)) * 100 );
+ }
+ return ( fn(Math.floor(num)) + dec + " " + add );
+ }
+
+ function sort(a,b){
+ last_col = -1;
+ var sort_reverse = false;
+ var sortWay = a.getAttribute("sort_" + b);
+ if (sortWay && sortWay == "forward") {
+ a.setAttribute("sort_" + b, "reverse");
+ sort_reverse = true;
+ }
+ else {
+ a.setAttribute("sort_" + b, "forward");
+ }
+ var c,d,e,f,g,h,i;
+ c=a.rows.length;
+ if(c<1){ return; }
+ d=a.rows[1].cells.length;
+ e=1;
+ var j=new Array(c);
+ f=0;
+ for(h=e;h
n[b]) ? true : false;
+ var lt = (m[b] parseInt(n[b], 10) ? true : false; lt = parseInt(m[b], 10) < parseInt(n[b], 10) ? true : false; }
+ if (sort_reverse) {gt = (!gt); lt = (!lt);}
+ if(gt){
+ j[i+1]=m;
+ j[i]=n;
+ l=true;
+ }
+ }
+ else{
+ if(lt){
+ j[i+1]=m;
+ j[i]=n;
+ l=true;
+ }
+ }
+ }
+ if(l===false){
+ break;
+ }
+ }
+ f=e;
+ for(h=0;h 1024) { add = "KB"; mul= 1024; }
+ if (num > (1024*1024)) { add = "MB"; mul= 1024*1024; }
+ if (num > (1024*1024*1024)) { add = "GB"; mul= 1024*1024*1024; }
+ if (num > (1024*1024*1024*1024)) { add = "TB"; mul= 1024*1024*1024*1024; }
+ num = num / mul;
+ if (add != "b" && num < 10) {
+ dec = "." + Math.floor( (num - Math.floor(num)) * 100 );
+ }
+ return ( Math.floor(num) + dec + " " + add );
+}
+
+// Hue, Saturation and Lightness to Red, Green and Blue:
+function quokka_internal_hsl2rgb (h,s,l)
+{
+ var min, sv, switcher, fract, vsf;
+ h = h % 1;
+ if (s > 1) s = 1;
+ if (l > 1) l = 1;
+ var v = (l <= 0.5) ? (l * (1 + s)) : (l + s - l * s);
+ if (v === 0)
+ return { r: 0, g: 0, b: 0 };
+
+ min = 2 * l - v;
+ sv = (v - min) / v;
+ var sh = (6 * h) % 6;
+ switcher = Math.floor(sh);
+ fract = sh - switcher;
+ vsf = v * sv * fract;
+
+ switch (switcher)
+ {
+ case 0: return { r: v, g: min + vsf, b: min };
+ case 1: return { r: v - vsf, g: v, b: min };
+ case 2: return { r: min, g: v, b: min + vsf };
+ case 3: return { r: min, g: v - vsf, b: v };
+ case 4: return { r: min + vsf, g: min, b: v };
+ case 5: return { r: v, g: min, b: v - vsf };
+ }
+ return {r:0, g:0, b: 0};
+}
+
+// RGB to Hex conversion
+function quokka_internal_rgb2hex(r, g, b) {
+ return "#" + ((1 << 24) + (Math.floor(r) << 16) + (Math.floor(g) << 8) + Math.floor(b)).toString(16).slice(1);
+}
+
+
+// Generate color list used for charts
+var colors = [];
+var rgbs = []
+var numColorRows = 6;
+var numColorColumns = 20;
+for (var x=0;x 8 ? (Math.random()) : (rnd[0]/255), y > 8 ? (0.75+(y*0.05)) : (rnd[1]/255), y > 8 ? (0.42 + (y*0.05*(x/numColorRows))) : (0.1 + rnd[2]/512));
+
+ // Light (primary) color:
+ var hex = quokka_internal_rgb2hex(color.r*255, color.g*255, color.b*255);
+
+ // Darker variant for gradients:
+ var dhex = quokka_internal_rgb2hex(color.r*131, color.g*131, color.b*131);
+
+ // Medium variant for legends:
+ var mhex = quokka_internal_rgb2hex(color.r*200, color.g*200, color.b*200);
+
+ colors.push([hex, dhex, color, mhex]);
+ }
+}
+
+
+/* Function for drawing pie diagrams
+ * Example usage:
+ * quokkaCircle("canvasName", [ { title: 'ups', value: 30}, { title: 'downs', value: 70} ] );
+ */
+
+function quokkaCircle(id, tags, opts) {
+ // Get Canvas object and context
+ var canvas = document.getElementById(id);
+ var ctx=canvas.getContext("2d");
+
+ // Calculate the total value of the pie
+ var total = 0;
+ var k;
+ for (k in tags) {
+ tags[k].value = Math.abs(tags[k].value);
+ total += tags[k].value;
+ }
+
+
+
+ // Draw the empty pie
+ var begin = 0;
+ var stop = 0;
+ var radius = (canvas.height*0.75)/2;
+ ctx.clearRect(0, 0, canvas.width, canvas.height);
+ ctx.beginPath();
+ ctx.shadowBlur = 6;
+ ctx.shadowOffsetX = 6;
+ ctx.shadowOffsetY = 6;
+ ctx.shadowColor = "#555";
+ ctx.lineWidth = (opts && opts.hires) ? 6 : 2;
+ ctx.strokeStyle = "#222";
+ ctx.arc((canvas.width-140)/2,canvas.height/2,radius, 0, Math.PI * 2);
+ ctx.closePath();
+ ctx.stroke();
+ ctx.fill();
+ ctx.shadowBlur = 0;
+ ctx.shadowOffsetY = 0;
+ ctx.shadowOffsetX = 0;
+
+
+ // Draw a title if set:
+ if (opts && opts.title) {
+ ctx.font= (opts && opts.hires) ? "28px Sans-Serif" : "15px Sans-Serif";
+ ctx.fillStyle = "#000000";
+ ctx.textAlign = "center";
+ ctx.fillText(opts.title,(canvas.width-140)/2, (opts && opts.hires) ? 30:15);
+ ctx.textAlign = "left";
+ }
+
+ ctx.beginPath();
+ var posY = 50;
+ var left = 120 + ((canvas.width-140)/2) + ((opts && opts.hires) ? 40 : 25)
+ for (k in tags) {
+ var val = tags[k].value;
+ stop = stop + (2 * Math.PI * (val / total));
+
+ // Make a pizza slice
+ ctx.beginPath();
+ ctx.lineCap = 'round';
+ ctx.arc((canvas.width-140)/2,canvas.height/2,radius,begin,stop);
+ ctx.lineTo((canvas.width-140)/2,canvas.height/2);
+ ctx.closePath();
+ ctx.lineWidth = 0;
+ ctx.stroke();
+
+ // Add color gradient
+ var grd=ctx.createLinearGradient(0,canvas.height*0.2,0,canvas.height);
+ grd.addColorStop(0,colors[k % colors.length][1]);
+ grd.addColorStop(1,colors[k % colors.length][0]);
+ ctx.fillStyle = grd;
+ ctx.fill();
+ begin = stop;
+
+ // Make color legend
+ ctx.fillRect(left, posY-((opts && opts.hires) ? 15 : 10), (opts && opts.hires) ? 14 : 7, (opts && opts.hires) ? 14 : 7);
+
+ // Add legend text
+ ctx.shadowColor = "rgba(0,0,0,0)"
+ ctx.font= (opts && opts.hires) ? "22px Sans-Serif" : "12px Sans-Serif";
+ ctx.fillStyle = "#000";
+ ctx.fillText(tags[k].title + " (" + Math.floor(val) + (opts && opts.pct ? "%" : "") + ")",left+20,posY);
+
+ posY += (opts && opts.hires) ? 28 : 14;
+ }
+
+}
+
+
+/* Function for drawing line charts
+ * Example usage:
+ * quokkaLines("myCanvas", ['Line a', 'Line b', 'Line c'], [ [x1,a1,b1,c1], [x2,a2,b2,c2], [x3,a3,b3,c3] ], { stacked: true, curve: false, title: "Some title" } );
+ */
+function quokkaLines(id, titles, values, options, sums) {
+ var canvas = document.getElementById(id);
+ var ctx=canvas.getContext("2d");
+ // clear the canvas first
+ ctx.clearRect(0, 0, canvas.width, canvas.height);
+
+
+
+
+ ctx.lineWidth = 0.25;
+ ctx.strokeStyle = "#000000";
+
+ var lwidth = 300;
+ var lheight = 75;
+ wspace = (options && options.hires) ? 110 : 55;
+ var rectwidth = canvas.width - lwidth - wspace;
+ var stack = options ? options.stack : false;
+ var curve = options ? options.curve : false;
+ var title = options ? options.title : null;
+ var spots = options ? options.points : false;
+ var noX = options ? options.nox : false;
+ var verts = options ? options.verts : true;
+ if (noX) {
+ lheight = 0;
+ }
+
+
+ // calc rectwidth if titles are large
+ var nlwidth = 0
+ for (var k in titles) {
+ ctx.font= (options && options.hires) ? "24px Sans-Serif" : "12px Sans-Serif";
+ ctx.fillStyle = "#00000";
+ var x = parseInt(k)
+ if (!noX) {
+ x = x + 1;
+ }
+ var sum = 0
+ for (var y in values) {
+ sum += values[y][x]
+ }
+ var t = titles[k] + (!options.nosum ? " (" + ((sums && sums[k]) ? sums[k] : sum.toFixed(0)) + ")" : "");
+ var w = ctx.measureText(t).width + 48;
+ if (w > lwidth && w > nlwidth) {
+ nlwidth = w
+ }
+ if (nlwidth > 0) {
+ rectwidth -= nlwidth - lwidth
+ lwidth = nlwidth
+ }
+ }
+
+ // Draw a border
+ ctx.lineWidth = 0.5;
+ ctx.strokeRect((wspace*0.75), 30, rectwidth, canvas.height - lheight - 40);
+
+ // Draw a title if set:
+ if (title != null) {
+ ctx.font= (options && options.hires) ? "24px Sans-Serif" : "15px Sans-Serif";
+ ctx.fillStyle = "#00000";
+ ctx.textAlign = "center";
+ ctx.fillText(title,rectwidth/2, 20);
+ }
+
+ // Draw legend
+ ctx.textAlign = "left";
+ var posY = 50;
+ for (var k in titles) {
+ var x = parseInt(k)
+ if (!noX) {
+ x = x + 1;
+ }
+ var sum = 0
+ for (var y in values) {
+ sum += values[y][x]
+ }
+
+ var title = titles[k] + (!options.nosum ? (" (" + ((sums && sums[k]) ? sums[k] : sum.toFixed(0)) + ")") : "");
+ if (options && options.lastsum) {
+ title = titles[k] + " (" + values[values.length-1][x].toFixed(0) + ")";
+ }
+ ctx.fillStyle = colors[k % colors.length][3];
+ ctx.fillRect(wspace + rectwidth + 75 , posY-((options && options.hires) ? 18:9), (options && options.hires) ? 20:10, (options && options.hires) ?20:10);
+
+ // Add legend text
+ ctx.font= (options && options.hires) ? "24px Sans-Serif" : "14px Sans-Serif";
+ ctx.fillStyle = "#00000";
+ ctx.fillText(title,canvas.width - lwidth + ((options && options.hires) ? 100:60), posY);
+
+ posY += (options && options.hires) ? 30:15;
+ }
+
+ // Find max and min
+ var max = null;
+ var min = 0;
+ var stacked = null;
+ for (x in values) {
+ var s = 0;
+ for (y in values[x]) {
+ if (y > 0 || noX) {
+ s += values[x][y];
+ if (max === null || max < values[x][y]) {
+ max = values[x][y];
+ }
+ if (min === null || min > values[x][y]) {
+ min = values[x][y];
+ }
+ }
+ }
+ if (stacked === null || stacked < s) {
+ stacked = s;
+ }
+ }
+ if (min == max) max++;
+ if (stack) {
+ min = 0;
+ max = stacked;
+ }
+
+
+ // Set number of lines to draw and each step
+ var numLines = 5;
+ var step = (max-min) / (numLines+1);
+
+ // Prettify the max value so steps aren't ugly numbers
+ if (step %1 != 0) {
+ step = (Math.round(step+0.5));
+ max = step * (numLines+1);
+ }
+
+ // Draw horizontal lines
+
+ for (x = -1; x <= numLines; x++) {
+ ctx.beginPath();
+ var y = 30 + (((canvas.height-40-lheight) / (numLines+1)) * (x+1));
+ ctx.moveTo(wspace*0.75, y);
+ ctx.lineTo(wspace*0.75 + rectwidth, y);
+ ctx.lineWidth = 0.25;
+ ctx.stroke();
+
+ // Add values
+ ctx.font= (options && options.hires) ? "20px Sans-Serif" : "12px Sans-Serif";
+ ctx.fillStyle = "#000000";
+
+ var val = Math.round( ((max-min) - (step*(x+1))) );
+ if (options && options.traffic) {
+ val = quokka_fnmb(val);
+ }
+ ctx.textAlign = "left";
+ ctx.fillText( val,canvas.width - lwidth - 20, y+8);
+ ctx.textAlign = "right";
+ ctx.fillText( val,wspace-32, y+8);
+ ctx.closePath();
+ }
+
+
+
+ // Draw vertical lines
+ var sx = 1
+ var numLines = values.length-1;
+ var step = (canvas.width - lwidth - wspace*0.75) / values.length;
+ while (step < 24) {
+ step *= 2
+ sx *= 2
+ }
+
+
+ if (verts) {
+ ctx.beginPath();
+ for (var x = 1; x < values.length; x++) {
+ if (x % sx == 0) {
+ var y = (wspace*0.75) + (step * (x/sx));
+ ctx.moveTo(y, 30);
+ ctx.lineTo(y, canvas.height - 10 - lheight);
+ ctx.lineWidth = 0.25;
+ ctx.stroke();
+ }
+ }
+ ctx.closePath();
+ }
+
+
+
+ // Some pre-calculations of steps
+ var step = (rectwidth) / (values.length > 1 ? values.length-1:1);
+
+ // Draw X values if noX isn't set:
+ if (noX != true) {
+ ctx.beginPath();
+ for (var i = 0; i < values.length; i++) {
+ zz = 1
+ var x = (wspace*0.75) + ((step) * i);
+ var y = canvas.height - lheight + 5;
+ if (i % sx == 0) {
+ ctx.translate(x, y);
+ ctx.moveTo(0,0);
+ ctx.lineTo(0,-15);
+ ctx.stroke();
+ ctx.rotate(45*Math.PI/180);
+ ctx.textAlign = "left";
+ var val = values[i][0];
+ if (val.constructor.toString().match("Date()")) {
+ val = val.toDateString();
+ }
+ ctx.fillText(val.toString(), 0, 0);
+ ctx.rotate(-45*Math.PI/180);
+ ctx.translate(-x,-y);
+ }
+ }
+ ctx.closePath();
+
+ }
+
+
+
+
+ // Draw each line
+ var stacks = [];
+ var pstacks = [];
+ for (k in values) { if (k > 0) { stacks[k] = 0; pstacks[k] = canvas.height - 40 - lheight; }}
+
+ for (k in titles) {
+ var maxY = 0, minY = 99999;
+ ctx.beginPath();
+ var color = colors[k % colors.length][0];
+ var f = parseInt(k) + 1;
+ if (noX) {
+ f = parseInt(k);
+ }
+ var value = values[0][f];
+ var step = rectwidth / numLines;
+ var x = (wspace*0.75);
+ var y = (canvas.height - 10 - lheight) - (((value-min) / (max-min)) * (canvas.height - 40 - lheight));
+ var py = y;
+ if (stack) {
+ stacks[0] = stacks[0] ? stacks[0] : 0
+ y -= stacks[0];
+ pstacks[0] = stacks[0];
+ stacks[0] += (((value-min) / (max-min)) * (canvas.height - 40 - lheight));
+ }
+
+ // Draw line
+ ctx.moveTo(x, y);
+ var pvalY = y;
+ var pvalX = x;
+ for (var i in values) {
+ if (i > 0) {
+ x = (wspace*0.75) + (step*i);
+ var f = parseInt(k) + 1;
+ if (noX == true) {
+ f = parseInt(k);
+ }
+ value = values[i][f];
+ y = (canvas.height - 10 - lheight) - (((value-min) / (max-min)) * (canvas.height - 40 - lheight));
+ if (stack) {
+ y -= stacks[i];
+ pstacks[i] = stacks[i];
+ stacks[i] += (((value-min) / (max-min)) * (canvas.height - 40- lheight));
+ }
+ if (y > maxY) maxY = y;
+ if (y < minY) minY = y;
+ // Draw curved lines??
+ /* We'll do: (x1,y1)-----(x1.5,y1)
+ * |
+ * (x1.5,y2)-----(x2,y2)
+ * with a quadratic beizer thingy
+ */
+ if (curve) {
+ ctx.bezierCurveTo((pvalX + x) / 2, pvalY, (pvalX + x) / 2, y, x, y);
+ pvalX = x;
+ pvalY = y;
+ }
+ // Nope, just draw straight lines
+ else {
+ ctx.lineTo(x, y);
+ }
+ if (spots) {
+ ctx.fillStyle = color;
+ ctx.translate(x-2, y-2);
+ ctx.rotate(-45*Math.PI/180);
+ ctx.fillRect(-2,1,4,4);
+ ctx.rotate(45*Math.PI/180);
+ ctx.translate(-x+2, -y+2);
+ }
+ }
+ }
+
+ ctx.lineWidth = 4;
+ ctx.strokeStyle = color;
+ ctx.stroke();
+
+
+ if (minY == maxY) maxY++;
+
+ // Draw stack area
+ if (stack) {
+ ctx.globalAlpha = 0.65;
+ for (i in values) {
+ if (i > 0) {
+ var f = parseInt(k) + 1;
+ if (noX == true) {
+ f = parseInt(k);
+ }
+ x = (wspace*0.75) + (step*i);
+ value = values[i][f];
+ y = (canvas.height - 10 - lheight) - (((value-min) / (max-min)) * (canvas.height - 40 - lheight));
+ y -= stacks[i];
+ }
+ }
+ var pvalY = y;
+ var pvalX = x;
+ if (y > maxY) maxY = y;
+ if (y < minY) minY = y;
+ for (i in values) {
+ var l = values.length - i - 1;
+ x = (wspace*0.75) + (step*l);
+ y = canvas.height - 10 - lheight - pstacks[l];
+ if (y > maxY) maxY = y;
+ if (y < minY) minY = y;
+ if (curve) {
+ ctx.bezierCurveTo((pvalX + x) / 2, pvalY, (pvalX + x) / 2, y, x, y);
+ pvalX = x;
+ pvalY = y;
+ }
+ else {
+ ctx.lineTo(x, y);
+ }
+ }
+ ctx.lineTo((wspace*0.75), py - pstacks[0]);
+ ctx.lineWidth = 0;
+ var grad = ctx.createLinearGradient(0, minY, 0, maxY);
+ grad.addColorStop(0.25, colors[k % colors.length][0])
+ grad.addColorStop(1, colors[k % colors.length][1])
+ ctx.strokeStyle = colors[k % colors.length][0];
+ ctx.fillStyle = grad;
+ ctx.fill();
+ ctx.fillStyle = "#000"
+ ctx.strokeStyle = "#000"
+ ctx.globalAlpha = 1;
+ }
+ ctx.closePath();
+ }
+
+ // draw feather
+ base_image = new Image();
+ base_image.src = '';
+ base_image.onload = function(){
+ ctx.globalAlpha = 0.15
+ ctx.drawImage(base_image, (canvas.width/2) - 64 - (lwidth/2), (canvas.height/2) - 128);
+ ctx.globalAlpha = 1
+ }
+}
+
+
+
+/* Function for drawing line charts
+ * Example usage:
+ * quokkaLines("myCanvas", ['Line a', 'Line b', 'Line c'], [ [x1,a1,b1,c1], [x2,a2,b2,c2], [x3,a3,b3,c3] ], { stacked: true, curve: false, title: "Some title" } );
+ */
+function quokkaBars(id, titles, values, options) {
+ var canvas = document.getElementById(id);
+ var ctx=canvas.getContext("2d");
+ // clear the canvas first
+ ctx.clearRect(0, 0, canvas.width, canvas.height);
+ var lwidth = 150;
+ var lheight = 75;
+ var stack = options ? options.stack : false;
+ var astack = options ? options.astack : false;
+ var curve = options ? options.curve : false;
+ var title = options ? options.title : null;
+ var noX = options ? options.nox : false;
+ var verts = options ? options.verts : true;
+ if (noX) {
+ lheight = 0;
+ }
+
+
+
+ // Draw a border
+ ctx.lineWidth = 0.5;
+ ctx.strokeRect(25, 30, canvas.width - lwidth - 40, canvas.height - lheight - 40);
+
+ // Draw a title if set:
+ if (title != null) {
+ ctx.font="15px Arial";
+ ctx.fillStyle = "#000";
+ ctx.textAlign = "center";
+ ctx.fillText(title,(canvas.width-lwidth)/2, 15);
+ }
+
+ // Draw legend
+ ctx.textAlign = "left";
+ var posY = 50;
+ for (var k in titles) {
+ var x = parseInt(k)
+ if (!noX) {
+ x = x + 1;
+ }
+ var title = titles[k];
+ if (title && title.length > 0) {
+ ctx.fillStyle = colors[k % colors.length][0];
+ ctx.fillRect(canvas.width - lwidth + 20, posY-10, 10, 10);
+
+ // Add legend text
+ ctx.font="12px Arial";
+ ctx.fillStyle = "#000";
+ ctx.fillText(title,canvas.width - lwidth + 40, posY);
+
+ posY += 15;
+ }
+
+
+ }
+
+ // Find max and min
+ var max = null;
+ var min = 0;
+ var stacked = null;
+ for (x in values) {
+ var s = 0;
+ for (y in values[x]) {
+ if (y > 0 || noX) {
+ s += values[x][y];
+ if (max == null || max < values[x][y]) {
+ max = values[x][y];
+ }
+ if (min == null || min > values[x][y]) {
+ min = values[x][y];
+ }
+ }
+ }
+ if (stacked == null || stacked < s) {
+ stacked = s;
+ }
+ }
+ if (min == max) {
+ max++;
+ }
+ if (stack) {
+ min = 0;
+ max = stacked;
+ }
+
+
+ // Set number of lines to draw and each step
+ var numLines = 5;
+ var step = (max-min) / (numLines+1);
+
+ // Prettify the max value so steps aren't ugly numbers
+ if (step %1 != 0) {
+ step = (Math.round(step+0.5));
+ max = step * (numLines+1);
+ }
+
+ // Draw horizontal lines
+ for (x = numLines; x >= 0; x--) {
+
+ var y = 30 + (((canvas.height-40-lheight) / (numLines+1)) * (x+1));
+ ctx.moveTo(25, y);
+ ctx.lineTo(canvas.width - lwidth - 15, y);
+ ctx.lineWidth = 0.25;
+ ctx.stroke();
+
+ // Add values
+ ctx.font="10px Arial";
+ ctx.fillStyle = "#000";
+ ctx.textAlign = "right";
+ ctx.fillText( Math.round( ((max-min) - (step*(x+1))) * 100 ) / 100,canvas.width - lwidth + 12, y-4);
+ ctx.fillText( Math.round( ((max-min) - (step*(x+1))) * 100 ) / 100,20, y-4);
+ }
+
+
+ // Draw vertical lines
+ var sx = 1
+ var numLines = values.length-1;
+ var step = (canvas.width - lwidth - 40) / values.length;
+ while (step < 24) {
+ step *= 2
+ sx *= 2
+ }
+
+
+ if (verts) {
+ ctx.beginPath();
+ for (var x = 1; x < values.length; x++) {
+ if (x % sx == 0) {
+ var y = 35 + (step * (x/sx));
+ ctx.moveTo(y, 30);
+ ctx.lineTo(y, canvas.height - 10 - lheight);
+ ctx.lineWidth = 0.25;
+ ctx.stroke();
+ }
+ }
+ }
+
+
+
+ // Some pre-calculations of steps
+ var step = (canvas.width - lwidth - 48) / values.length;
+ var smallstep = (step / titles.length) - 2;
+
+ // Draw X values if noX isn't set:
+ if (noX != true) {
+ ctx.beginPath();
+ for (var i = 0; i < values.length; i++) {
+ smallstep = (step / (values[i].length-1)) - 2;
+ zz = 1
+ var x = 35 + ((step) * i);
+ var y = canvas.height - lheight + 5;
+ if (i % sx == 0) {
+ ctx.translate(x, y);
+ ctx.moveTo(0,0);
+ ctx.lineTo(0,-15);
+ ctx.stroke();
+ ctx.rotate(45*Math.PI/180);
+ ctx.textAlign = "left";
+ var val = values[i][0];
+ if (val.constructor.toString().match("Date()")) {
+ val = val.toDateString();
+ }
+ ctx.fillText(val.toString(), 0, 0);
+ ctx.rotate(-45*Math.PI/180);
+ ctx.translate(-x,-y);
+ }
+ }
+
+ }
+
+
+
+
+ // Draw each line
+ var stacks = [];
+ var pstacks = [];
+
+ for (k in values) {
+ smallstep = (step / (values[k].length)) - 2;
+ stacks[k] = 0;
+ pstacks[k] = canvas.height - 40 - lheight;
+ var beginX = 0;
+ for (i in values[k]) {
+ if (i > 0 || noX) {
+ var z = parseInt(i);
+ var zz = z;
+ if (!noX) {
+ z = parseInt(i) + 1;
+ zz = z - 2;
+ if (z > values[k].length) {
+ break;
+ }
+ }
+ var value = values[k][i];
+ var title = titles[i];
+ var color = colors[zz % colors.length][1];
+ var fcolor = colors[zz % colors.length][2];
+ if (values[k][2] && values[k][2].toString().match(/^#.+$/)) {
+ color = values[k][2]
+ fcolor = values[k][2]
+ smallstep = (step / (values[k].length-2)) - 2;
+ }
+ var x = ((step) * k) + ((smallstep+2) * zz) + 5;
+ var y = canvas.height - 10 - lheight;
+ var mdiff = (max-min);
+ mdiff = (mdiff == 0) ? 1 : mdiff;
+ var height = ((canvas.height - 40 - lheight) / (mdiff)) * value * -1;
+ var width = smallstep - 2;
+ if (width <= 1) {
+ width = 1
+ }
+ if (stack) {
+ width = step - 10;
+ y -= stacks[k];
+ stacks[k] -= height;
+ x = (step * k) + 4;
+ if (astack) {
+ y = canvas.height - 10 - lheight;
+ }
+ }
+
+
+ // Draw bar
+ ctx.beginPath();
+ ctx.lineWidth = 2;
+ ctx.strokeStyle = color;
+ ctx.strokeRect(27 + x, y, width, height);
+ var alpha = 0.75
+ if (fcolor.r) {
+ ctx.fillStyle = 'rgba('+ [parseInt(fcolor.r*255),parseInt(fcolor.g*255),parseInt(fcolor.b*255),alpha].join(",") + ')';
+ } else {
+ ctx.fillStyle = fcolor;
+ }
+ ctx.fillRect(27 + x, y, width, height);
+
+ }
+ }
+
+
+ }
+}
+
+
+]==]
+
+
+status_css = [[
+ html {
+ font-size: 14px;
+ position: relative;
+ background: #272B30;
+ }
+
+ body {
+ background-color: #272B30;
+ color: #000;
+ margin: 0 auto;
+ min-height: 100%;
+ font-family: Arial, Helvetica, sans-serif;
+ font-weight: normal;
+ }
+
+ .navbarLeft {
+ background: linear-gradient(to bottom, #F8A900 0%,#D88900 100%);
+ width: 200px;
+ height: 30px;
+ padding-top: 2px;
+ font-size: 1.35rem;
+ color: #FFF;
+ border-bottom: 2px solid #000;
+ float: left;
+ text-align: center;
+ }
+
+ .navbarRight {
+ background: linear-gradient(to bottom, #EFEFEF 0%,#EEE 100%);
+ width: calc(100% - 240px);
+ height: 28px;
+ color: #333;
+ border-bottom: 2px solid #000;
+ float: left;
+ font-size: 1.3rem;
+ padding-top: 4px;
+ text-align: left;
+ padding-left: 40px;
+ }
+
+ .wrapper {
+ width: 100%;
+ float: left;
+ background: #33363F;
+ min-height: calc(100% - 80px);
+ position: relative;
+ }
+
+ .serverinfo {
+ float: left;
+ width: 200px;
+ height: calc(100% - 34px);
+ background: #293D4C;
+ }
+
+ .skey {
+ background: rgba(30,30,30,0.3);
+ color: #C6E7FF;
+ font-weight: bold;
+ padding: 2px;
+ }
+
+ .sval {
+ padding: 2px;
+ background: rgba(30,30,30,0.3);
+ color: #FFF;
+ font-size: 0.8rem;
+ border-bottom: 1px solid rgba(200,200,200,0.2);
+ }
+
+ .charts {
+ padding: 0px;
+ width: calc(100% - 220px);
+ max-width: 1000px;
+ min-height: 100%;
+ margin: 0px auto;
+ position: relative;
+ float: left;
+ margin-left: 20px;
+ }
+
+ pre, code {
+ font-family: "Courier New", Courier, monospace;
+ }
+
+ strong {
+ font-weight: bold;
+ }
+
+ q, em, var {
+ font-style: italic;
+ }
+ /* h1 */
+ /* ====================== */
+ h1 {
+ padding: 0.2em;
+ margin: 0;
+ border: 1px solid #405871;
+ background-color: inherit;
+ color: #036;
+ text-decoration: none;
+ font-size: 22px;
+ font-weight: bold;
+ }
+
+ /* h2 */
+ /* ====================== */
+ h2 {
+ padding: 0.2em 0 0.2em 0.7em;
+ margin: 0 0 0.5em 0;
+ text-decoration: none;
+ font-size: 18px;
+ font-weight: bold;
+ text-align: center;
+ }
+
+ #modules {
+ margin-top:20px;
+ display:none;
+ width:400px;
+ }
+
+ .servers {
+
+ width: 1244px;
+ background: #EEE;
+ }
+
+ tr:nth-child(odd) {
+ background: #F6F6F6;
+ }
+ tr:nth-child(even) {
+ background: #EBEBEB;
+ }
+ td {
+ padding: 2px;
+ }
+ table {
+ border: 1px solid #333;
+ padding: 0px;
+ margin: 5px;
+ min-width: 360px;
+ background: #999;
+ font-size: 0.8rem;
+ }
+
+ canvas {
+ background: #FFF;
+ margin: 3px;
+ text-align: center;
+ padding: 2px;
+ border-radius: 10px;
+ border: 1px solid #999;
+ }
+
+ .canvas_wide {
+ position: relative;
+ width: 65%;
+ }
+ .canvas_narrow {
+ position: relative;
+ width: 27%;
+ }
+
+ a {
+ color: #FFA;
+ }
+
+ .statsbox {
+ border-radius: 3px;
+ background: #3C3E47;
+ min-width: 150px;
+ height: 60px;
+ float: left;
+ margin: 15px;
+ padding: 10px;
+ }
+
+ .btn {
+ background: linear-gradient(to bottom, #72ca72 0%,#55bf55 100%);
+ border-radius: 5px;
+ color: #FFF;
+ text-decoration: none;
+ padding-top: 6px;
+ padding-bottom: 6px;
+ padding-left: 3px;
+ padding-right: 3px;
+ font-weight: bold;
+ text-shadow: 1px 1px rgba(0,0,0,0.4);
+ margin: 12px;
+ float: left;
+ clear: none;
+ }
+
+ .infobox_wrapper {
+ float: left;
+ min-width: 200px;
+ margin: 10px;
+ }
+ .infobox_title {
+ border-top-left-radius: 4px;
+ border-top-right-radius: 4px;
+ background: #FAB227;
+ color: #FFF;
+ border: 2px solid #FAB227;
+ border-bottom: none;
+ font-weight: bold;
+ text-align: center;
+ width: 100%;
+ }
+ .infobox {
+ background: #222222;
+ border: 2px solid #FAB227;
+ border-top: none;
+ color: #EFEFEF;
+ border-bottom-left-radius: 4px;
+ border-bottom-right-radius: 4px;
+ float: left;
+ width: 100%;
+
+ }
+
+
+ .serverinfo ul {
+ margin: 0px;
+ padding: 0px;
+ margin-top: 20px;
+ list-style: none;
+ }
+
+ .serverinfo ul li .btn {
+ width: calc(100% - 8px);
+ margin: 0px;
+ border: 0px;
+ border-radius: 0px;
+ padding: 0px;
+ padding-top: 8px;
+ padding-left: 8px;
+ height: 24px;
+ background: rgba(0,0,0,0.2);
+ border-bottom: 1px solid rgba(100,100,100,0.3);
+ }
+
+ .serverinfo ul li:nth-child(1) {
+ border-top: 1px solid rgba(100,100,100,0.3);
+ }
+ .serverinfo ul li .btn.active {
+ background: rgba(30,30,50,0.2);
+ border-left: 4px solid #27FAB2;
+ padding-left: 4px;
+ color: #FFE;
+ }
+
+ .serverinfo ul li .btn:hover {
+ background: rgba(50,50,50,0.15);
+ border-left: 4px solid #FAB227;
+ padding-left: 4px;
+ color: #FFE;
+ }
+]]
\ No newline at end of file