A Highcharts wrapper for Angular (version 4 and newer)
- lazily-load the Highcharts library (60+ KB gzipped) and any additional modules from CDN or your own distribution
- take advantage of the Highcharts official typings
- automatically resize the charts to fulfill its container
- easy installation, configuration, usage and testing, with examples
yarn add @howtimeflies/ngx-highcharts
yarn add highcharts --dev
or
npm install --save @howtimeflies/ngx-highcharts
npm install --save-dev highcharts
Import the module and set the configuration options in app.module.ts
import { HighchartsConfig, HighchartsModule } from '@howtimeflies/ngx-highcharts'
const config: HighchartsConfig = {
cdnBaseUrl: 'https://code.highcharts.com',
scriptName: 'highcharts.js',
delayToExecuteModulesCode: 200,
maxDelayToResizeContainer: 10000,
globalOptions: {
lang: {
drillUpText: 'Drill-Up'
}
}
}
@NgModule({
imports: [
...,
HighchartsModule
],
providers: [
{ provide: HighchartsConfig, useValue: config }
],
bootstrap: [...]
})
export class AppModule { }
The default configuration options are
public static defaultConfig: HighchartsConfig = {
// The base url of the Highcharts library on CDN
cdnBaseUrl: 'https://unpkg.com/highcharts',
// The javascript file name of the Highcharts library. We could set it to 'highcharts.src.js' in debugging.
scriptName: 'highcharts.js',
// The delay in milliseconds to execute the additional Highcharts modules.
delayToExecuteModulesCode: 500,
// The max delay in milliseconds to wait the browser to draw the chart.
// Only after the chart element is available on DOM, we could resize it to fulfill its container
maxDelayToResizeContainer: 10000,
// The global options of Highcharts as listed in https://api.highcharts.com/highcharts/lang
globalOptions: {}
}
If you would like to use the default configuration options, simply provide an empty object to HighchartsConfig
{ provide: HighchartsConfig, useValue: {} }
The missing options in your custom config would be set with the ones from the default config.
Add a <ngx-highchart>
element to the HTML template. It must be wrapped in a parent container.
<div class="chart-container">
<ngx-highchart [options]="options" (load)="chart = $event.chart"></ngx-highchart>
</div>
Set the chart options in the typescript code, and get the chart object in the load
event handler.
import { Chart, Options } from 'highcharts'
export class ChartComponent {
public options: Options = {
chart: { type: 'area' },
series:[
{
name: 'name',
type: 'area',
data: []
}
],
...
}
public chart: Chart
}
Update the chart data dynamically.
this.chart.series[0].setData([1, 0, 3, null, 3, 1, 2, 1])
this.chart.addSeries({
type: 'area',
name: 'John',
data: [0, 1, 4, 4, 5, 2, 3, 7]
})
If some additional highcharts modules are required, set them in the modules
attribute.
<div class="chart-container">
<ngx-highchart [options]="options" [modules]="['modules/drilldown', 'highcharts-more']"
(load)="onLoad($event)"></ngx-highchart>
</div>
Set event handlers to the chart
public onLoad(evt: {chart: Chart; highcharts: typeof Highcharts}) {
this.chart = evt.chart
evt.chart.series[0].setData(this.data)
evt.highcharts.addEvent(evt.chart, 'drilldown', it => this.onDrillDown(it))
}
private onDrillDown(evt) {
const data = this.drillDownData.find(it => it.name === evt.point.name)
this.chart.addSeriesAsDrilldown(evt.point, data)
}
Install the Highcharts library as a dev dependency: yarn add highcharts --dev
or npm install --save-dev highcharts
Verify the chart data and event handler
import * as Highcharts from 'highcharts'
describe(`Drill-down Chart Component`, () => {
let comp: DrilldownChartComponent
let fixture: ComponentFixture<DrilldownChartComponent>
// add the required module
require('highcharts/modules/drilldown.src')(Highcharts)
beforeEach(async(() => {
fixture = TestBed.configureTestingModule({
declarations: [DrilldownChartComponent],
// import the module for testing
imports: [HighchartsTestingModule]
}).createComponent(DrilldownChartComponent)
fixture.detectChanges()
comp = fixture.componentInstance
}))
it(`should display the data on chart`, () => {
// retrieve the data from the chart object
const data = comp.chart.series[0].data.map((it: Highcharts.DataPoint) => [it.name, it.y])
// verify the data
expect(data).toContainEqual(['IE', 56.33])
expect(data).toContainEqual(['Chrome', 24.03])
})
it(`should drill down to a pie`, () => {
let data = null
// spy on the expecting behavior
spyOn(comp.chart, 'addSeriesAsDrilldown').and.callFake((x, drillDownData) => data = drillDownData)
const point = comp.chart.series[0].data[0]
// simulate an event on the chart object
Highcharts.fireEvent(comp.chart, 'drilldown', {point})
// verify the data
expect(data.name).toEqual('IE')
expect(data.data).toContainEqual(['v11.0', 24.13])
})
})
With Angular CLI's default ng build
, the "highcharts" library is bundled into the production build since it is imported in the project for its typings. angular-cli issue
To work it around, please extend ng build
with @angular-builders/custom-webpack:browser
, and then mark 'highcharts' as external in the extra webpack config.
Please check the example project for details.