Highcharts : selection - highcharts
I made a barchart with dates. Currently, to remove the dates, you must click on the dates in the legend. I wish delete multiple dates at once.
Let's take an example. My dates range from 1960 to 2015. If I would have on my chart the dates from 2000 to 2005, so I would like that if you click on 1960 and 1999, then all the dates between are deleted / hidden. It goes faster when many dates.
How to do ?
Here is my code :
$(function () {
$('#container').highcharts({
chart: {
type: 'column'
},
colors:[
'#7cb5ec',
'#434348',
'#90ed7d',
'#f7a35c',
'#8085e9',
'#f15c80',
'#e4d354',
'#2b908f',
'#f45b5b',
'#91e8e1',
'#DA70D6',
'#1E90FF',
'#E0F000'
],
legend:{
itemStyle:{
fontSize:'10px',
font:'10pt',
color:'#000000'
}
},
title:{
style:{
fontSize:'0px'
}
},
subtitle:{
style:{
fontSize:'0px'
}
},
xAxis: {
categories: [
'',
],
},
yAxis: {
min: 15,
title: {
text: 'Number',
style:{
fontSize:'0px'
}
}
},
tooltip: {
shared: false,
useHTML: true
},
plotOptions: {
column: {
pointPadding: 0.2,
borderWidth: 0
}
},
series: [{
name: '2004',
data: [49.9],
},
{
name: '2005',
data: [83.6]
},
{
name: '2006',
data: [48.9]
},
{
name: '2007',
data: [69.1]
},
{
name: '2008',
data: [83.6]
},
{
name: '2009',
data: [40.9]
},
{
name: '2010',
data: [69.9]
},
{
name: '2011',
data: [83]
},
{
name: '2012',
data: [28.9]
},
{
name: '2013',
data: [40.9]
},
{
name: '2014',
data: [81.6]
},
{
name: '2015',
data: [24.9]
},
{
name: '2016',
data: [46.4]
}]
});
});
Thanks
Rather than having each year as its own series, I would recommend that you place the years as the values in your x-axis (as categories) and then use Highcharts' native zoom function to allow users to select a certain date range.
I've updated your fiddle with some changes, which I explain below: https://jsfiddle.net/brightmatrix/hr28s27L/
First, in your chart option, I added zoomType: 'x' to allow users to use their mouse to zoom in on specific years. The value x means they can only zoom horizontally (across), which makes for a "cleaner" interaction.
chart: {
type: 'column',
zoomType: 'x'
},
Second, I updated your x-axis with the years as the categories/values:
xAxis: {
categories: ['2004','2005','2006','2007','2008','2009','2010','2011','2012','2013','2014','2015','2016']
},
Lastly, I updated your series to show just the data points, which are then plotted on the y-axis:
series: [{
name: 'data by year',
data: [49.9,83.6,48.9,69.1,83.6,40.9,69.9,83,28.9,40.9,81.6,24.9,46.4]
}]
Here is what happens when a user clicks and drags their mouse cursor:
The shaded box is what they are selecting. Once they release the cursor, the chart will automatically adjust to show only what they chose. A "Reset Zoom" button will appear at the top right to allow them to go back to all years:
Update (July 19, 2016): I did some research on adding a simple slider element to this chart that allows users to choose from a range of years.
An updated version of my fiddle that shows this example chart with sliders can be found here: https://jsfiddle.net/brightmatrix/uvat8u05/.
The code to handle the sliders is shown below. I discovered that explicitly setting the slider values to integers using the parseInt() function solved a couple of problems:
You can correctly compare the slider values to make sure users can't choose an end year that is earlier than the start year.
The setExtremes() function correctly shows a single year when users choose the same start and end year.
All x-axis labels are shown at all times. Before I added parseInt(), some labels disappeared when users chose different values in the sliders.
Please be sure to read the comments I've left in both the HTML and Javascript panes, as I've included additional details on why I made certain code decisions.
// on change handler for both sliders
$('.mySlider').bind('change', function(e) {
e.preventDefault();
var chart = $('#container').highcharts();
// convert the slider values to integers to make sure setExtremes() works as expected
var slider1Val = parseInt($('input[name="slider1"]').val());
var slider2Val = parseInt($('input[name="slider2"]').val());
if (slider2Val < slider1Val) {
// warn the user if they try to choose an end year that is later than the start year
alert("You can't choose an end year that is earlier than your start year.");
} else {
// use setExtremes to set the x-axis ranges based on the values in the sliders
chart.xAxis[0].setExtremes(slider1Val, slider2Val);
}
});
I hope this information is helpful for you.
Related
highcharts lollipop/dumbbell chart change position/colour of positive and negative marker values
I am trying to create a lollipop chart where some values are negative, and some are positive, either side of zero, like this: image However the chart seemingly only allows the marker to be on the "high" end of the chart. Is there a way to control which end has the marker ? I'm using these chart options: chartOptions: { chart: { type: 'dumbbell', inverted: true }, title: { text: 'lollipop chart' }, xAxis: { type: 'category' }, yAxis: { title: { text: 'value' } }, series: [{ name: 'value', data: [{ name: 'a', low:0, high:13, }, { name: 'b', low:0, high:26, },{ name: 'c', low:-43, high:0 },{ name: 'd', low:-83, high:0 },{ name: 'e', low:0, high:113 }] }] }
The series which should be used in this case is a lollipop, but it seems that the array of objects configuration doesn't work. I reported it on the Highcharts Github issue channel, where you can follow this thread: https://github.com/highcharts/highcharts/issues/13202 As a solution in the dumbell series I suggest to find those graphics in the render callback and hide them. Demo: https://jsfiddle.net/BlackLabel/3o7acsbt/ events: { render() { let chart = this; chart.series[0].points.forEach(p => { if (p.low >= 0){ p.lowerGraphic.hide() } else { p.upperGraphic.hide() } }) } } API: https://api.highcharts.com/highcharts/chart.events.render EDIT Under the GitHub link, the workaround showed up. Please check it, maybe will be fit better to your requirement.
Highcharts - Performance issue using chart with x values date and TIME
I am plotting a chart with ‘y’ axis being numeric values and ‘x’ axis being date time values. I was able to process a JSON object of 3000 items. However, because of our business rules, we also need to display the time: hours + minutes in the chart’s tooltips. This changed everything! Now the chart takes about 15 seconds to plot 2000 records. This is unacceptable. I can see clearly that when I remove the time part of my Date object the charts works perfectly. It is just when added the hours and minutes that performance gets affected. Trying different things I realized that your charts should support the amount of data I am using since it is not massive. We love the charts but performance is key for us to continue using your products. Can you help me with this issue? Please check this fiddle so you can understand my problem. Feel free to remove the hours and minutes variables from the DateUtc creation: https://jsfiddle.net/17a3jry9/7/ Thanks in advance! var pointStart = new Date(); var data = [{"date":"2017-11-08","time":"1712","perc":10},{"date":"2017-11-08","time":"1608","perc":10},{"date":"2017-11-08","time":"1506","perc":10},{"date":"2017-11-08","time":"1408","perc":10},{"date":"2017-11-08","time":"1309","perc":10},{"date":"2017-11-08","time":"1207","perc":10},{"date":"2017-11-08","time":"1110","perc":10},{"date":"2017-11-08","time":"1003","perc":10},{"date":"2017-11-08","time":"0910","perc":10},{"date":"2017-11-08","time":"0810","perc":10},{"date":"2017-11-08","time":"0708","perc":10},{"date":"2017-11-09","time":"1710","perc":10},{"date":"2017-11-09","time":"1604","perc":10},{"date":"2017-11-09","time":"1510","perc":10},{"date":"2017-11-09","time":"1406","perc":10},{"date":"2017-11-09","time":"1310","perc":10},{"date":"2017-11-09","time":"1205","perc":10},{"date":"2017-11-09","time":"1107","perc":10},{"date":"2017-11-09","time":"1010","perc":10},{"date":"2017-11-09","time":"0912","perc":10},{"date":"2017-11-09","time":"0806","perc":10}{"date":"2018-10-25","time":"0709","perc":10},{"date":"2018-10-25","time":"1009","perc":10},{"date":"2018-10-25","time":"1208","perc":10},{"date":"2018-10-25","time":"1309","perc":10},{"date":"2018-10-25","time":"1410","perc":10},{"date":"2018-10-25","time":"1510","perc":10},{"date":"2018-10-25","time":"1702","perc":10},{"date":"2018-10-26","time":"1409","perc":10},{"date":"2018-10-26","time":"0710","perc":10},{"date":"2018-10-26","time":"1505","perc":10},{"date":"2018-10-26","time":"1704","perc":10},{"date":"2018-10-29","time":"0708","perc":10},{"date":"2018-10-29","time":"1007","perc":10},{"date":"2018-10-29","time":"1208","perc":10},{"date":"2018-10-29","time":"1406","perc":10},{"date":"2018-10-29","time":"1509","perc":10},{"date":"2018-10-29","time":"1610","perc":10},{"date":"2018-10-30","time":"0710","perc":10},{"date":"2018-10-30","time":"1010","perc":10},{"date":"2018-10-30","time":"1207","perc":10},{"date":"2018-10-30","time":"1409","perc":10},{"date":"2018-10-30","time":"1510","perc":10},{"date":"2018-10-30","time":"1709","perc":10},{"date":"2018-10-31","time":"0708","perc":10},{"date":"2018-10-31","time":"1009","perc":10},{"date":"2018-10-31","time":"1206","perc":10},{"date":"2018-10-31","time":"1409","perc":10},{"date":"2018-10-31","time":"1509","perc":10},{"date":"2018-10-31","time":"1708","perc":10},{"date":"2018-11-01","time":"0707","perc":10},{"date":"2018-11-01","time":"1007","perc":10},{"date":"2018-11-01","time":"1108","perc":10},{"date":"2018-11-01","time":"1250","perc":10},{"date":"2018-11-01","time":"1509","perc":10},{"date":"2018-11-01","time":"1407","perc":10},{"date":"2018-11-01","time":"1709","perc":10},{"date":"2018-11-02","time":"0708","perc":10},{"date":"2018-11-02","time":"1007","perc":10},{"date":"2018-11-02","time":"1108","perc":10},{"date":"2018-11-02","time":"1210","perc":10},{"date":"2018-11-02","time":"1407","perc":10},{"date":"2018-11-02","time":"1509","perc":10},{"date":"2018-11-02","time":"1608","perc":10},{"date":"2018-11-02","time":"1715","perc":10},{"date":"2018-11-05","time":"0707","perc":10},{"date":"2018-11-05","time":"1007","perc":10},{"date":"2018-11-05","time":"1209","perc":10},{"date":"2018-11-05","time":"1408","perc":10},{"date":"2018-11-05","time":"1509","perc":10},{"date":"2018-11-05","time":"1611","perc":10},{"date":"2018-11-05","time":"1715","perc":10},{"date":"2018-11-06","time":"0708","perc":10},{"date":"2018-11-06","time":"1007","perc":10},{"date":"2018-11-06","time":"1201","perc":10},{"date":"2018-11-06","time":"1410","perc":10},{"date":"2018-11-06","time":"1510","perc":10},{"date":"2018-11-06","time":"1709","perc":10},{"date":"2018-11-07","time":"0708","perc":10},{"date":"2018-11-07","time":"1007","perc":10},{"date":"2018-11-07","time":"1209","perc":10},{"date":"2018-11-07","time":"1307","perc":10},{"date":"2018-11-07","time":"1410","perc":10},{"date":"2018-11-07","time":"1506","perc":10},{"date":"2018-11-07","time":"1708","perc":10},{"date":"2018-11-08","time":"0707","perc":10},{"date":"2018-11-08","time":"1009","perc":10},{"date":"2018-11-08","time":"1207","perc":10},{"date":"2018-11-08","time":"1309","perc":10},{"date":"2018-11-08","time":"1412","perc":10},{"date":"2018-11-08","time":"1508","perc":10}]; var newSeries = data.map(function (key) { var utilDate = new Date(key.date); var hours = parseInt(key.time.slice(0, 2)); var minutes = parseInt(key.time.slice(2, 4)); return { x: Date.UTC(utilDate.getFullYear(), utilDate.getMonth(), utilDate.getDate(), hours, minutes), y: key.perc, key: key.id }; }); Highcharts.stockChart('container', { xAxis: { type: 'datetime', title: { text: 'Date' } }, yAxis: { title: { text: 'Util (%)' }, min: 0 }, tooltip: { headerFormat: '<b>{point.x: %A, %b %e, %I:%M %p}</b><br>' //pointFormat: '{point.x:%e. %b}: {point.y:.2f} m' }, colors: ['teal', 'red'], // Define the data points. All series have a dummy year // of 1970/71 in order to be compared on the same x axis. Note // that in JavaScript, months start at 0 for January, 1 for February etc. series: [{ name: "Util", data: newSeries }], title: { text: 'Util Movement' }, subtitle: { text: "" }, plotOptions: { series: { cursor: 'pointer', turboThreshold: 10000, point: { events: { } }, label: { connectorAllowed: false }, pointStart: pointStart.getFullYear() } }, responsive: { rules: [{ condition: { maxWidth: 200 }, chartOptions: { legend: { layout: 'horizontal', align: 'center', verticalAlign: 'bottom' } } }] } });
Please take a look to the browser console, you have information there about error: Highcharts error #15: www.highcharts.com/errors/15. This means that you have unsorted data, so you need to sort them: newSeries.sort(function(a, b){ return a.x - b.x }); Live demo: https://jsfiddle.net/BlackLabel/onq0Lurx/
How to show two specific data points from tabular data in Highcharts?
I am trying to plot a chart with only two data points from a CSV file that holds many more data points with a datetime x-Axis. In the end the user should be able to compare two years of his choice out of the whole dataset. From the API documentation I know that I can pick a range of data points from a CSV with startRow and endRow: https://api.highcharts.com/highcharts/data.startRow But that would only plot one specific point, as you can see in my fiddle. Is there any other way to programmatically show specific points? Highcharts.chart('container', { chart: { type: 'column', polar: false, }, title: { text: '' }, subtitle: { text: '' }, data: { csv: document.getElementById('csv').innerHTML, startRow: 3, endRow: 4, googleSpreadsheetKey: false, googleSpreadsheetWorksheet: false }, series: [{ name: 'val' }], yAxis: [{ title: { text: '' }, opposite: true, labels: { align: 'right', reserveSpace: true } }], xAxis: [{ type: 'datetime', opposite: false }], pane: { background: [] }, responsive: { rules: [] }, legend: { title: '' }, plotOptions: { series: { animation: false } } }); Edit: I forgot to mention, that i only want to load the CSV once. After that the user should be able to select/update data points without reloading the data. For showing ranges of values dynamically, I used Axis min and max settings like this: $(this).highcharts().update({ xAxis:{ min: Date.UTC(selectedStart,0,0), max: Date.UTC(selectedEnd,11,31) } }); The whole dataset is loaded once and the chart axis gets updated on user interaction. Now I am looking for something similar just not for a range but a comparison of two values off of the whole dataset.
Highcharts data module provides a function called parsed that allows you to modify the fetched data programmatically before it's applied to the chart. From Highcharts API (https://api.highcharts.com/highcharts/data.parsed): parsed: function A callback function to access the parsed columns, the two-dimentional input data array directly, before they are interpreted into series data and categories. Return false to stop completion, or call this.complete() to continue async. Live demo: http://jsfiddle.net/gh/get/library/pure/highcharts/highcharts/tree/master/samples/highcharts/data/parsed/
Show data using highchart bar graph where categories are dynamic date which comes with data
Show data using highchart bar graph where categories are dynamic date which comes with data. My code to show data on graph: function highchartDate(Male,Female) { Highcharts.chart('container', { chart: { type: 'column' }, xAxis: { type: 'datetime', tickInterval: 24 * 3600 * 1000, // one day startOnTick: true, endOnTick: true, dateTimeLabelFormats: { day: '%e of %b' } }, tooltip: { shared: true }, series: [{ name: 'Day Shift', data: Male }, { name: 'Night Shift', data: Female }] }); }; where Male and Female are the array of arrays like Male = [[1508803200000, 41][1509235200000, 44][1509408000000, 44]] Female =[[1508803200000, 41][1509235200000, 44][1509408000000, 44]] when i run this code on fiddle it works fine but not in project I noticed the reason but don't get the way to make it worked in code. Can any one help me in this getting the data displayed on graph properly. this is the Link where you get the code which run fine if i put the value in data seperated with commas like [[1508803200000, 41],[1509235200000, 44],[1509408000000, 44]] .this works fine : https://jsfiddle.net/Pookool/k1qua683/1/ #Kamil Kulig can you help me in this or any one else
highstock charts - button for date range
in my application I use angular Highstock for charts. I have a lot of data (many millions), and now i want to put the existing buttons for set date range (1week, 1 month, 3 months, 1year, all), but also to click on any button to send a new request to the server to not accumulating data. Example, if pressed button "one week", that only the server retrieves the data for one week, if you click on a button to one month, to take data for a single month. I do not want to call unnecessary data, and that the server slows down, if I'm looking for data for one month, and the server comes back to me all data. Here is the code. Thanks in advance > https://jsfiddle.net/tf7pr1ft/
Thnx to #Grzegorz Blachliński i fix this. here is code $scope.chartConfig1 = { xAxis: { ordinal: false, //za dodat date range postavit events i range selectors events : { afterSetExtremes: getDateRange //here i add my custom function } }, yAxis: { plotLines: [{ color: '#FF0000', width: 1, value: 11.50, label: {text: '11.50'} }] }, options: { chart: { zoomType: 'x', backgroundColor: 'rgba(255, 255, 255, 1)', polar: true, type: 'line', borderRadius: 5 }, legend: { enabled: true }, rangeSelector: { selected : 0, //here i defined default range enabled: true, inputStyle: { color: 'black' } }, navigator: { enabled: true } }, series: [], title: { text: 'Solar and Battery voltage average' }, useHighStocks: true }, $scope.chartConfig1.series.push({ id: 1, name: "Solar voltage average", data: $scope.data[0], tooltip: { valueDecimals: 2 } }, { id: 2, name: "Battery voltage average", data: $scope.data[1], tooltip: { valueDecimals: 2 } } ); and here is my custom function getDateRange = function (e) { var date1 = new Date(e.min); var date2 = new Date(e.max); var timeDiff = Math.abs(date2.getTime() - date1.getTime()); var diffDays = Math.ceil(timeDiff / (1000 * 3600 * 24)); $http.get(serviceBase + 'aaaaa/aaaaaa/' + $stateParams.klupaID + '/aaaaaaa?days=' + diffDays, function(data){ //ukoliko buden dodavat još koji config obavezno dodat broj koliko ih ima na i<=8 for (i=1; i<=8; i++){ $scope.chartConfigString = 'chartConfig' + i; $scope.chartConfigString.series[0].setData(data); $scope.chartConfigString.hideLoading(); } }); }; Now i have another problems, i want to have four selectors (buttons), 1months, 3months, 1year and ALL. I set default range, when open app, on 1month, and here is problem. Buttons for another range is disabled, beacause i get data from url only for one month. I want to have enabled all buttons for select any range. Thnx