How to add custom values to nodes of plotly graph - javascript
I'm trying to add values to nodes of plotly graph but there is an issue.
Code:
<html>
<head>
<script src="https://cdn.plot.ly/plotly-latest.min.js"></script>
<style>
.graph-container {
display: flex;
justify-content: center;
align-items: center;
}
.main-panel {
width: 70%;
float: left;
}
.side-panel {
width: 30%;
background-color: lightgray;
min-height: 300px;
overflow: auto;
float: right;
}
</style>
</head>
<body>
<div class="graph-container">
<div id="myDiv" class="main-panel"></div>
<div id="lineGraph" class="side-panel"></div>
</div>
<script>
var nodes = [
{ x: 0, y: 0, z: 0, value: [1, 2, 3] },
{ x: 1, y: 1, z: 1, value: [4, 5, 6] },
{ x: 2, y: 0, z: 2, value: [7, 8, 9] },
{ x: 3, y: 1, z: 3, value: [10, 11, 12] },
{ x: 4, y: 0, z: 4, value: [13, 14, 15] }
];
var edges = [
{ source: 0, target: 1 },
{ source: 1, target: 2 },
{ source: 2, target: 3 },
{ source: 3, target: 4 }
];
var x = [];
var y = [];
var z = [];
for (var i = 0; i < nodes.length; i++) {
x.push(nodes[i].x);
y.push(nodes[i].y);
z.push(nodes[i].z);
}
var xS = [];
var yS = [];
var zS = [];
var xT = [];
var yT = [];
var zT = [];
for (var i = 0; i < edges.length; i++) {
xS.push(nodes[edges[i].source].x);
yS.push(nodes[edges[i].source].y);
zS.push(nodes[edges[i].source].z);
xT.push(nodes[edges[i].target].x);
yT.push(nodes[edges[i].target].y);
zT.push(nodes[edges[i].target].z);
}
var traceNodes = {
x: x, y: y, z: z,
mode: 'markers',
marker: { size: 12, color: 'red' },
type: 'scatter3d',
text: [0, 1, 2, 3, 4],
hoverinfo: 'text',
hoverlabel: {
bgcolor: 'white'
},
customdata: nodes.map(function(node) {
if (node.value !== undefined)
return node.value;
}),
type: 'scatter3d'
};
var layout = {
margin: { l: 0, r: 0, b: 0, t: 0 }
};
Plotly.newPlot('myDiv', [traceNodes], layout, { displayModeBar: false });
document.getElementById("myDiv").on("plotly_click", function(data){
var nodeIndex = data.points[0].pointIndex;
var values = nodes[nodeIndex].value;
console.log(values);
Plotly.newPlot('lineGraph', [{ x: [0, 1, 2], y: values, type: 'scatter3d'}], {
margin: { t: 0 }
});
});
</script>
</body>
</html>
var values = nodes[nodeIndex].value; is not able to access the node values.
Suggestions on how to fix this will be really helpful.
EDIT0:
<html>
<head>
<script src="https://cdn.plot.ly/plotly-1.58.5.min.js"></script>
<style>
.graph-container {
display: flex;
justify-content: center;
align-items: center;
}
.main-panel {
width: 70%;
float: left;
}
.side-panel {
width: 30%;
background-color: lightgray;
min-height: 300px;
overflow: auto;
float: right;
}
</style>
</head>
<body>
<div class="graph-container">
<div id="myDiv" class="main-panel"></div>
<div id="lineGraph" class="side-panel"></div>
</div>
<script>
var nodes = [
{ x: 0, y: 0, z: 0, value: [1, 2, 3] },
{ x: 1, y: 1, z: 1, value: [4, 5, 6] },
{ x: 2, y: 0, z: 2, value: [7, 8, 9] },
{ x: 3, y: 1, z: 3, value: [10, 11, 12] },
{ x: 4, y: 0, z: 4, value: [13, 14, 15] }
];
var edges = [
{ source: 0, target: 1 },
{ source: 1, target: 2 },
{ source: 2, target: 3 },
{ source: 3, target: 4 }
];
var x = [];
var y = [];
var z = [];
for (var i = 0; i < nodes.length; i++) {
x.push(nodes[i].x);
y.push(nodes[i].y);
z.push(nodes[i].z);
}
var xS = [];
var yS = [];
var zS = [];
var xT = [];
var yT = [];
var zT = [];
for (var i = 0; i < edges.length; i++) {
xS.push(nodes[edges[i].source].x);
yS.push(nodes[edges[i].source].y);
zS.push(nodes[edges[i].source].z);
xT.push(nodes[edges[i].target].x);
yT.push(nodes[edges[i].target].y);
zT.push(nodes[edges[i].target].z);
}
var traceNodes = {
x: x, y: y, z: z,
mode: 'markers',
marker: { size: 12, color: 'red' },
type: 'scatter3d',
text: [0, 1, 2, 3, 4],
hoverinfo: 'text',
hoverlabel: {
bgcolor: 'white'
},
customdata: nodes.map(function(node) {
if (node.value !== undefined)
return node.value;
}),
type: 'scatter3d'
};
var layout = {
margin: { l: 0, r: 0, b: 0, t: 0 }
};
Plotly.newPlot('myDiv', [traceNodes], layout, { displayModeBar: false });
const ymax = Math.max(...nodes.map(n => n.value).flat());
document.getElementById("myDiv").on("plotly_click", function(data){
var nodeIndex = data.points[0].pointNumber;
var values = nodes[nodeIndex].value;
console.log(values);
Plotly.newPlot('lineGraph', [{ x: [0, 1, 2], y: values, type: 'scatter3d'}], {
margin: { t: 0 },
yaxis: {autorange: false, range: [0, ymax + 1]}
});
});
</script>
</body>
</html>
I made the corrections suggested in the answer below but the I am still not able to see the line graph when nodes in the main panel are clicked.
Use pointNumber instead of pointIndex. You also need to fix the type of the second plot, for which I also suggest to set a fixed axis range (otherwise the axis will reset for each selected point) :
// max y value for the line plot
const ymax = Math.max(...nodes.map(n => n.value).flat());
document.getElementById('myDiv').on('plotly_click', function(data){
var nodeIndex = data.points[0].pointNumber;
var values = nodes[nodeIndex].value;
Plotly.newPlot('lineGraph', [{
type: 'scatter',
mode: 'lines',
x: [0, 1, 2],
y: values
}], {
margin: { t: 0 },
yaxis: {autorange: false, range: [0, ymax + 1]}
});
});
For reference, here is the event data structure for 3d plots :
{
points: [{
curveNumber: 2, // index in data of the trace associated with the selected point
pointNumber: 2, // index of the selected point
x: 5, // x value
y: 600, // y value
z: 12, // z value
data: {/* */}, // ref to the trace as sent to Plotly.newPlot associated with the selected point
fullData: {/* */}, // ref to the trace including all of the default attributes
xaxis: {/* */}, // ref to x-axis object (i.e layout.xaxis) associated with the selected point
yaxis: {/* */} // ref to y-axis object " "
zaxis: {/* */} // ref to z-axis object " "
}, {
/* similarly for other selected points */
}]
}
One last thing, don't use plotly-latest.js or its minified version anymore. While still referenced here and there in the documentation, they are stuck at version 1.58.5.
Please note that as of v2 the "plotly-latest" outputs will no longer be updated on the CDN, and will stay at the last v1 patch v1.58.5. Therefore, to use the CDN with plotly.js v2 and higher, you must specify an exact plotly.js version.
Here is the complete code :
<html>
<head>
<script src="https://cdn.plot.ly/plotly-2.18.1.min.js"></script>
<style>
.graph-container {
display: flex;
justify-content: center;
align-items: center;
}
.main-panel {
width: 70%;
float: left;
}
.side-panel {
width: 30%;
background-color: lightgray;
min-height: 300px;
overflow: auto;
float: right;
}
</style>
</head>
<body>
<div class="graph-container">
<div id="myDiv" class="main-panel"></div>
<div id="lineGraph" class="side-panel"></div>
</div>
<script>
var nodes = [
{ x: 0, y: 0, z: 0, value: [1, 2, 3] },
{ x: 1, y: 1, z: 1, value: [4, 5, 6] },
{ x: 2, y: 0, z: 2, value: [7, 8, 9] },
{ x: 3, y: 1, z: 3, value: [10, 11, 12] },
{ x: 4, y: 0, z: 4, value: [13, 14, 15] }
];
var edges = [
{ source: 0, target: 1 },
{ source: 1, target: 2 },
{ source: 2, target: 3 },
{ source: 3, target: 4 }
];
var x = [];
var y = [];
var z = [];
for (var i = 0; i < nodes.length; i++) {
x.push(nodes[i].x);
y.push(nodes[i].y);
z.push(nodes[i].z);
}
var xS = [];
var yS = [];
var zS = [];
var xT = [];
var yT = [];
var zT = [];
for (var i = 0; i < edges.length; i++) {
xS.push(nodes[edges[i].source].x);
yS.push(nodes[edges[i].source].y);
zS.push(nodes[edges[i].source].z);
xT.push(nodes[edges[i].target].x);
yT.push(nodes[edges[i].target].y);
zT.push(nodes[edges[i].target].z);
}
var traceNodes = {
x: x, y: y, z: z,
mode: 'markers',
marker: { size: 12, color: 'red' },
type: 'scatter3d',
text: [0, 1, 2, 3, 4],
hoverinfo: 'text',
hoverlabel: {
bgcolor: 'white'
},
customdata: nodes.map(function(node) {
if (node.value !== undefined)
return node.value;
}),
type: 'scatter3d'
};
var layout = {
margin: { l: 0, r: 0, b: 0, t: 0 }
};
Plotly.newPlot('myDiv', [traceNodes], layout, { displayModeBar: false });
// max y value for the line plot
const ymax = Math.max(...nodes.map(n => n.value).flat());
document.getElementById('myDiv').on('plotly_click', function(data){
var nodeIndex = data.points[0].pointNumber;
var values = nodes[nodeIndex].value;
Plotly.newPlot('lineGraph', [{
type: 'scatter',
mode: 'lines',
x: [0, 1, 2],
y: values
}], {
margin: { t: 0 },
yaxis: {autorange: false, range: [0, ymax + 1]}
});
});
</script>
</body>
</html>
Related
Chartjs: Sum of each dataset on top
Having this data: const datasets = [ { label: "# of Votes x", data: [12, 19, 3, 5, 2, 3], fill: true, }, { label: "# of Votes y", data: [7, 3, 13, 18, 20, 6], fill: true, }, ]; Is it possible to add at the top of the graph the sum for that column? Something like that:
Yes, you can do that. 1. Preprocess your data to get the sum for each column let sums = [0, 0, 0, 0, 0, 0]; for (let i = 0; i < sums.length; i++) { for (let dataset of datasets) { sums[i] += dataset.data[i]; } } 2. Create a second x-axis with a new position options: { scales: { // ... x2: { position: 'top', labels: sums }, // ... } } Here is the result: const ctx = document.getElementById('myChart'); let datasets = [ { label: "# of Votes x", data: [12, 19, 3, 5, 2, 3], fill: true }, { label: "# of Votes y", data: [7, 3, 13, 18, 20, 6], fill: true } ]; let sums = [0, 0, 0, 0, 0, 0]; for (let i = 0; i < sums.length; i++) { for (let dataset of datasets) { sums[i] += dataset.data[i]; } } new Chart(ctx, { type: 'line', data: { datasets }, options: { scales: { x: { labels: ['Red', 'Blue', 'Yellow', 'Green', 'Purple', 'Orange'] }, x2: { position: 'top', labels: sums }, y: { beginAtZero: true } } } }); .chart-container { position: relative; height: 90vh; } <script src="https://cdn.jsdelivr.net/npm/chart.js"></script> <div class="chart-container"> <canvas id="myChart"></canvas> </div>
how to deleted an item with the same `x` and `y` but different key and also adds all the `g` with the same 'x' and 'y'
I have an array of objects: let input = [ { key: 0, x: 0, y: 0, g: 1 }, { key: 1, x: 0, y: 0, g: 2 }, { key: 2, x: 2, y: 0, g: 3 }, { key: 3, x: 0, y: 0, g: 3 }, { key: 4, x: 2, y: 2, g: 1 }, { key: 5, x: 2, y: 2, g: 2 }, ]; I mean I want to remove duplicate objects with the same x and y, and convert All of them to one, and also I want to add all the g which has same x and y. and I would like to sort them with the new key value. let result = [ { key: 0, x: 0, y: 0, g: 6 }, { key: 1, x: 2, y: 0, g: 3 }, { key: 2, x: 0, y: 0, g: 3 }, ]; is it possible to help me. I try by myself to remove but it doesn't work properly const set = new Set(json1.map((item) => JSON.stringify(item))); const uniq = [...set].map((item) => JSON.parse(item)); console.log(uniq)
This can be achieved with a standard 'group-by', grouping by a composite key of x and y and summing g. (Your stated expected output is not consistent with your description of your problem. And it is unclear if you want to retain the original keys or generate new ones) let input = [ { key: 0, x: 0, y: 0, g: 1 }, { key: 1, x: 0, y: 0, g: 2 }, { key: 2, x: 2, y: 0, g: 3 }, { key: 3, x: 0, y: 0, g: 3 }, { key: 4, x: 2, y: 2, g: 1 }, { key: 5, x: 2, y: 2, g: 2 }, ]; const result = Object.values( input.reduce((a, { key, x, y, g }) => (((a[`x_${x}_y_${y}`] ??= { key, x, y, g: 0 }).g += g), a), {}) ); console.log(result); .as-console-wrapper { max-height: 100% !important; top: 0; }
How to style specific gridlines differently in chartJs?
I am using chartJS to draw a bubble chart with an x axis running from -5 to +5 and a y axis running from -5 to +5. I was able to change the grid line styling using x: { grid: { borderDash: [2, 10], lineWidth: 1, color: `#000`, }, to get this result. my chart output so far I need to have the X and Y axis at 0 on both to be solid bold. Is there a way to style gridline at specific tick points? desired result is this... desired chart styling
The borderDash option is non scriptable so to achieve this behaviour you will need to use a custom plugin to draw over the default grid lines: const zeroZeroLines = { id: 'zeroZeroLines', beforeDatasetsDraw: (chart, args, opts) => { const { ctx, chartArea: { top, bottom, left, right }, scales: { x, y } } = chart; const color = opts.color || 'black'; const width = opts.width || 1; ctx.beginPath(); ctx.lineWidth = width; ctx.strokeStyle = color; ctx.moveTo(x.getPixelForValue(0), bottom); ctx.lineTo(x.getPixelForValue(0), top); ctx.moveTo(left, y.getPixelForValue(0)); ctx.lineTo(right, y.getPixelForValue(0)); ctx.stroke(); } } const options = { type: 'bubble', data: { datasets: [{ label: '# of Votes', data: [{ x: -4, y: 0, r: 4 }, { x: 1, y: -3, r: 10 }, { x: 3, y: 3, r: 20 }, { x: 0, y: 0, r: 20 }], backgroundColor: 'pink' }] }, options: { scales: { x: { min: -5, max: 5, grid: { borderDash: [2, 2] } }, y: { min: -5, max: 5, grid: { borderDash: [2, 2] } } }, plugins: { zeroZeroLines: { color: 'black', width: 1 } } }, plugins: [zeroZeroLines] } const ctx = document.getElementById('chartJSContainer').getContext('2d'); new Chart(ctx, options); <body> <canvas id="chartJSContainer" width="600" height="400"></canvas> <script src="https://cdnjs.cloudflare.com/ajax/libs/Chart.js/3.5.0/chart.js"></script> </body>
Chart.JS plugin, draw after animation
I'm trying to write a chart.js (v.2.7) plugin to show errorbars on my scatter / line plots. I managed to get the pixel positions from the chart and draw the errorbars on the canvas, but i cannot get the timing to look right. I want them to appear after the lines animate, then stay attached (& move with) if the datasets are hidden->shown. I've tried: The afterDraw / afterDataset(s)Draw / beforeDraw hooks: the error bars are already on the plot before the lines animate (as in example). When hidden->shown, the error bars are in place. afterRender / afterEvent hook: draws them after the animation finishes, but then redraws them everytime the datasets are hidden->shown (after a pause) beforeRender / any earlier hooks: gives no errorbars setTimout() on the draw function, or at various places inside it: does nothing sleep() before the draw function or other places: slows down the whole animation but the errorsbars are unaffected. I couldn't find a way to invoke plugin functions after animation or get to then via the options.animation.onComplete Is there a way to get the errorbars to behave as in the example, but the initial appearance occurs after the line animation (with a plugin?) var errorbarPlugin = { calcPoints: function(chartInstance, dataList){ var ds = chartInstance.data.datasets var meta = chartInstance.getDatasetMeta(0) var yScale = chartInstance.scales[meta.yAxisID]; var xScale = chartInstance.scales[meta.xAxisID]; var yList = []; var xList = []; for(var i = 0; i < dataList.length; i++){ var yValue = dataList[i].y var yPixel = yScale.getPixelForValue(yValue) yList.push(yPixel) var xValue = dataList[i].x var xPixel = xScale.getPixelForValue(xValue) xList.push(xPixel) } return {yList: yList, xList: xList} }, calcErrorbars: function(chartInstance, ds_num){ var ds = chartInstance.data.datasets var data_list = ds[ds_num].data var isHidden = ds[ds_num]._meta[Object.keys(chartInstance.data.datasets[ds_num]._meta)[0]].hidden; var yList = this.calcPoints(chartInstance, data_list).yList var xList = this.calcPoints(chartInstance, data_list).xList if(ds[ds_num].errors){ var errors = ds[ds_num].errors } else {errors = 0} return [xList, yList, errors, isHidden] }, drawErrorbars: function(chartInstance){ var ctx = chartInstance.chart.ctx var ds = chartInstance.data.datasets for(var ds_num = 0; ds_num < ds.length; ds_num++){ var errCalc = this.calcErrorbars(chartInstance, ds_num) var isHidden = errCalc[3] var yList = errCalc[1] var xList = errCalc[0] var errors = errCalc[2] var errWidth = 3 var capLen = 5 if(!isHidden){ for(var k = 0; k < xList.length; k++){ ctx.strokeStyle = "red" ctx.beginPath(); ctx.moveTo(xList[k], yList[k]-errors[k]); ctx.lineTo(xList[k], yList[k]+errors[k]); ctx.moveTo(xList[k]-capLen, yList[k]+errors[k]); ctx.lineTo(xList[k]+capLen, yList[k]+errors[k]); ctx.moveTo(xList[k]-capLen, yList[k]-errors[k]); ctx.lineTo(xList[k]+capLen, yList[k]-errors[k]); ctx.stroke() } } } }, afterDatasetsDraw: function(chartInstance) { this.drawErrorbars(chartInstance) }, } <html> <body> <script src="https://cdnjs.cloudflare.com/ajax/libs/Chart.js/2.7.2/Chart.js"></script> <script type="text/javascript" src="{% static 'js/charts/errorbarPlugin3.js' %}"></script> <div id="canvas-holder" class="col-sm-3"> <canvas id="chart-gamma" width="200" height="200"/></canvas> </div> <script defer> var gammaChartData = { datasets: [ { label: 'Red', data: [{x: 15, y: 30}, {x: 35, y: 17}, {x: 55, y: 37}, {x: 72, y: 45},], borderColor: "red", errors: [10, 28, 30, 34], }, ] } var ctx_gamma = document.getElementById("chart-gamma").getContext("2d"); window.onload = function() { var gamma_chart = new Chart(ctx_gamma, { type: 'scatter', data: gammaChartData, plugins: [errorbarPlugin], options: {showLines: true}, }); }; </script> </body> </html> Edit: shortened snippet by removing formatting and default options
Here's what I came up with (snippet below) The errorbars can either be fixed to the data or shown independantly, and animate in / are hidden / revealed with the data. The plugin can add x or y axis errors, repesented as bars (with/without caps) or ovals / circles (filled or transparent). "use strict"; var errorbarPlugin = { afterDraw: function (chart) { var type = chart.config.type; var plugConfig = chart.config.options.errorbarPlugin; if (plugConfig) { if (plugConfig.showErrors) { var showErrors = plugConfig.showErrors; } } else showErrors = true; if (showErrors !== false) { if (["line", "scatter"].includes(type)) { errorbarPlugin.scatterErrorbars(chart); } else if (type == "bar") { console.log("Bar charts not supported yet"); } } }, scatterErrorbars: function (chart) { var ctx = chart.ctx; var plugConfig = chart.config.options.errorbarPlugin; chart.data.datasets.forEach(function (dataset, i) { var ds = dataset; var meta = chart.getDatasetMeta(i); var showErrors; (ds.showErrors === false) ? showErrors = false : showErrors = true; var errWidth; (ds.errWidth) ? errWidth = ds.errWidth : errWidth = 1; var showCap; (ds.showCap) ? showCap = ds.showCap : showCap = true; var capLen; (ds.capLen) ? capLen = ds.capLen : capLen = 3; var errStyle; (ds.errStyle) ? errStyle = ds.errStyle : errStyle = "T"; var errFillColor; (ds.errFillColor) ? errFillColor = ds.errFillColor : errFillColor = "rgba(0,0,0,0)"; if (!meta.hidden && showErrors) { meta.data.forEach(function (element, index) { var x_point = element._model.x; var y_point = element._model.y; var errColor; (ds.errColor) ? errColor = ds.errColor : errColor = element._view.borderColor; var dataPoint = ds.data[index]; var yError; var xError; if (typeof (dataPoint) === "object" && 'r' in dataPoint) { yError = dataPoint.r; } else if (ds.errors) { yError = ds.errors[index]; } else { yError = null; } if (typeof (dataPoint) === "object" && dataPoint.e) { xError = dataPoint.e; } else if (ds.xErrors) { xError = ds.xErrors[index]; } else { xError = null; } var position = element.tooltipPosition(); if (errStyle == "circle") { ctx.beginPath(); ctx.arc(position.x, position.y, yError, 0, 2 * Math.PI, false); if (ds.hidden === true && meta.hidden === null) { ctx.strokeStyle = "rgba(0,0,0,0)"; ctx.fillStyle = "rgba(0,0,0,0)"; } else { ctx.strokeStyle = errColor; ctx.fillStyle = errFillColor; } console.log(meta.hidden); ctx.fill(); ctx.stroke(); } else if (errStyle == "oval" || errStyle == "ellipse") { if (xError) { var scaleFac = (xError) / yError; } else scaleFac = 10 / yError; ctx.beginPath(); ctx.save(); ctx.scale(scaleFac, 1); ctx.arc(position.x / scaleFac, position.y, yError, 0, 2 * Math.PI, false); ctx.restore(); if (ds.hidden === true && meta.hidden === null) { ctx.strokeStyle = "rgba(0,0,0,0)"; } else { ctx.strokeStyle = errColor; } ctx.stroke(); } else { ctx.beginPath(); ctx.moveTo(position.x, position.y - yError); ctx.lineTo(position.x, position.y + yError); if (xError) { ctx.moveTo(position.x - xError, position.y); ctx.lineTo(position.x + xError, position.y); } if (ds.hidden === true && meta.hidden === null) { ctx.strokeStyle = "rgba(0,0,0,0)"; } else { ctx.strokeStyle = errColor; } ctx.stroke(); if (showCap) { ctx.beginPath(); ctx.moveTo(position.x - capLen, position.y - yError); ctx.lineTo(position.x + capLen, position.y - yError); ctx.moveTo(position.x - capLen, position.y + yError); ctx.lineTo(position.x + capLen, position.y + yError); if (xError) { ctx.moveTo(position.x - xError, position.y - capLen); ctx.lineTo(position.x - xError, position.y + capLen); ctx.moveTo(position.x + xError, position.y - capLen); ctx.lineTo(position.x + xError, position.y + capLen); } ctx.stroke(); } } }); } }); } }; <!DOCTYPE html> <!--DOCTYPE html --> <html> <head> <script src='https://cdnjs.cloudflare.com/ajax/libs/Chart.js/2.6.0/Chart.min.js'></script> <body> <div style = "position:relative; width:60%;" > <div id="canvas-holder" class="col-sm-6"> <canvas id="chart-gamma" width="500" height="500"/></canvas> </div> <div id="canvas-holderbf2" class="col-sm-6"> <canvas id="chart-humid" width="500" height="500"/></canvas> </div> <script defer> Chart.defaults.global.legend.display = true Chart.defaults.global.legend.position = 'right' // Chart.defaults.global.legend.onHover = function(){} // Chart.defaults.global.legend.onClick = function(){} Chart.defaults.global.legend.labels.usePointStyle = true Chart.defaults.global.legend.labels.fontsize = 12 Chart.defaults.global.legend.labels.padding = 10 var gammaChartData = { datasets: [ { label: 'Eu', data: [{x: 5, y: 45}, {x: 10, y: 100}, {x: 25, y: 120}, {x: 50, y: 125}, {x: 100, y: 150}, {x: 120, y: 250},], borderColor: "red", //fillColor: "pink", errors: [15, 20, 30, 12, 10, 10], xErrors: [3, 7, 16, 12, 12, 30, 10], //hidden: true, errColor: "blue", errStyle: "circle", errFillColor: "pink", hidden: true, errWidth: 2, showCap: true, capLen: 3, showErrors: true, }, { label: 'Am', data: [{x: 15, y: 85, r: 14}, {x: 25, y: 37, r: 8}, {x: 62, y: 135, r: 44},], borderColor: "blue", errColor: "red", errStyle: "circle", showErrors: true, }, ] } var options_gamma = { animation: { duration: 1000, }, errorbarPlugin: { showErrors: true, marginsOfError: [100, 50, 10], }, elements: { line: { fill: false, borderWidth: 1, }, point: { radius: 0, pointStyle: 'circle', borderWidth: 1, hitRadius: 18, //size if should hover // hoverBorderWidth: 13, hoverRadius: 10, //size when hovered }, }, annotation: { annotations: [{ id: 'h-line-01', // optional type: 'line', mode: 'horizontal', scaleID: 'y-axis-0', value: '125', borderColor: 'red', borderDash: [2, 2], borderWidth: 2, label: { enabled: true, backgroundColor: 'rgba(255,255,255,1)', // Background color of label, default below //fontFamily: "sans-serif", // Font family of text, inherits from global fontStyle: "normal", // Font style of text, default "bold" fontSize: 12, // Font size of text, inherits from global fontColor: "red",// Font color of text, default below xPadding: 5,// Padding of label to add top/bottom, default below yPadding: 5,// Radius of label rectangle, default below cornerRadius: 10, // Anchor position of label on line, can be one of: top, bottom, left, right, center. Default below. position: "left", // Adjustment along x-axis (left-right) of label relative to above number (can be negative) // For horizontal lines positioned left or right, negative values move the label toward the edge, and negative values toward the center. xAdjust: 290, // Adjustment along y-axis (top-bottom) of label relative to above number (can be negative) // For vertical lines positioned top or bottom, negative values move the label toward the edge, and negative values toward the center. yAdjust: 0, // Whether the label is enabled and should be displayed // Text to display in label - default is null content: "Max" }, onClick: function(e) { // Fires when the user clicks this annotation on the chart (be sure to enable the event in the events array below). } }], }, responsive: true, showLines: true, hoverMode: 'single', // should always use single for a scatter chart legend: {}, scales: { yAxes: [{ display: true, position: 'left', id: 'y-axis-0', ticks: {min: 0, //beginAtZero:true, max: 200, //display: true, //fontColor: "black" }, scaleLabel: {display: true, labelString: 'Number'}, gridLines: {color: "black", //display: true, drawOnChartArea: false, zeroLineColor: "black", //drawTicks: true, }, }], xAxes: [{ display: true, type: 'linear', id: 'x-axis-0', position: 'bottom', ticks: {min: 0, max: 100, //display: true, //fontColor: "black", }, scaleLabel: {display: true, labelString: 'Volume'}, gridLines: {color: "black", zeroLineColor: "black", drawOnChartArea: false, }, }], }, } var ctx_gamma = document.getElementById("chart-gamma").getContext("2d"); var humidChartData = { datasets: [ { label: 'B errors', data: [{x: 5, y: 45}, {x: 10, y: 100}, {x: 25, y: 120}, {x: 50, y: 125}, {x: 100, y: 150}, {x: 120, y: 250},], borderColor: "green", errors: [15, 20, 30, 12, 10, 10], xErrors: [3, 7, 16, 12, 12, 30, 10], errStyle: "oval", showLine: false, errColor: "border", //pointBackgroundColor: "white", //pointBordercolor: "white", backgroundColor: "rgba(0,0,0,0)", hidden: true, errWidth: 2, showCap: true, capLen: 3, radius: 0, showErrors: true, }, { label: 'B trend', data: [{x: 5, y: 45}, {x: 10, y: 100}, {x: 25, y: 120}, {x: 50, y: 125}, {x: 100, y: 150}, {x: 120, y: 250},], borderColor: "green", errors: [15, 20, 30, 12, 10, 10], xErrors: [3, 7, 16, 12, 12, 30, 10], pointStyle: "line", showErrors: false, radius: 0, }, { label: 'B data', data: [{x: 5, y: 45}, {x: 10, y: 100}, {x: 25, y: 120}, {x: 50, y: 125}, {x: 100, y: 150}, {x: 120, y: 250},], borderColor: "green", backgroundColor: "green", errors: [15, 20, 30, 12, 10, 10], xErrors: [3, 7, 16, 12, 12, 30, 10], showErrors: false, showLine: false, }, { label: '', data: [], borderColor: "rgba(0,0,0,0)", backgroundColor: "rgba(0,0,0,0)", }, { label: 'C data', data: [{x: 15, y: 85, r: 14}, {x: 25, y: 37, r: 8}, {x: 62, y: 135, r: 44},], borderColor: "blue", backgroundColor: "rgba(0,0,0,0)", xErrors: [3, 7, 16, 12, 12, 30, 10], showLine: true, showErrors: true, }, ] } var options_humid = { hoverMode: 'single', elements: { line: { fill: false, borderWidth: 2, }, point: { radius: 3, pointStyle: 'circle', borderWidth: 1, hitRadius: 0, // hoverBorderWidth: 13, hoverRadius: 9, }, }, responsive: true, showLines: true, hoverMode: 'single', // should always use single for a scatter chart legend: { labels: { usePointStyle: true, // generateLabels: function() { } } }, scales: { yAxes: [{ display: true, position: 'left', id: 'y-axis-0', ticks: {min: 0, //beginAtZero:true, max: 300 }, scaleLabel: {display: true, labelString: 'Number'}, gridLines: {zeroLineColor: "black", }, }], xAxes: [{ display: true, type: 'linear', id: 'x-axis-0', position: 'bottom', ticks: {min: 0, max: 200 }, scaleLabel: {display: true, labelString: 'Month'}, gridLines: {zeroLineColor: "black", }, }], }, } var ctx_humid = document.getElementById("chart-humid").getContext("2d"); window.onload = function() { var humidChart = new Chart(ctx_humid, { type: 'line', data: humidChartData, plugins: [errorbarPlugin], options: options_humid, }); var gamma_chart = new Chart(ctx_gamma, { type: 'scatter', data: gammaChartData, plugins: [errorbarPlugin], options: options_gamma, }); }; </script> </div> </body> </html>
I had a similar issue with rendered text in plugins:[{ afterDatasetsDraw: function(chart, options) { var ctx = chart.ctx; ctx.font = Chart.defaults.global.defaultFontStyle; ctx.fillStyle = "#666666"; ctx.textAlign = 'center'; ctx.textBaseline = 'bottom'; chart.data.datasets.forEach(function(dataset, i) { var meta = chart.controller.getDatasetMeta(i); meta.data.forEach(function (bar, index) { ctx.fillText(Math.round(dataset.data[index]), bar._model.x, bar._model.y); }); }); } }] To fix, inside the meta.data.forEach(function (bar, index) { ... } loop, simply use: var position = bar.tooltipPosition(); ctx.fillText(Math.round(dataset.data[index]), position.x, position.y); (This is in #Yobmod's post, but it seems the key is to use the bar.tooltipPosition() location for the location of whatever is rendered.)
How to add date on x-axis in g.raphael line chart
I am using Raphael line chart ..But how i display date on x-axis. I tried below code. window.onload = function () { var r = Raphael("holder"), txtattr = { font: "12px 'Fontin Sans', Fontin-Sans, sans-serif" }; var lines = r.linechart(330, 250, 300, 220, [[5, 6, 7, 8, 9, 10, 11],[3, 4, 5, 6, 7, 8,10]], [[17, 32, 23, 15, 17, 11,13]], { nostroke: false, axis: "0 0 1 1", symbol: "circle", smooth: true }).hoverColumn(function () { this.tags = r.set(); for (var i = 0, ii = this.y.length; i < ii; i++) { this.tags.push(r.tag(this.x, this.y[i], this.values[i], 160, 10).insertBefore(this).attr([{ fill: "#fff" }, { fill: this.symbols[i].attr("fill") }])); } }, function () { this.tags && this.tags.remove(); }); lines.symbols.attr({ r: 6 }); }; By using above code its displaying number on x-axis.. But how to display date on X-axis.