Area chart
TIP
For core principles, concept description, API and changelog, refer to the D3 chart.
Basic usage
import React from 'react';
import { Chart } from '@semcore/ui/d3-chart';
function formatDate(value) {
const options = {
month: 'short' as const,
day: 'numeric' as const,
};
return new Intl.DateTimeFormat('en', options).format(value);
}
const Demo = () => {
return (
<Chart.Area
groupKey={'time'}
data={data}
plotWidth={500}
plotHeight={200}
tooltipValueFormatter={formatDate}
aria-label={'Area chart'}
/>
);
};
const date = new Date();
const data = Array(10)
.fill({})
.map((d, i) => {
return {
time: new Date(date.setDate(date.getDate() + 5)),
line: Math.random() * 10,
};
});
export default Demo;
Area
- You can draw a chart with areas using the
Area
component. Dots
are the dots on the line chart.- As with the Line chart, you can draw a polyline or a smoothed chart by passing the required method to the curve property.
import React from 'react';
import { Plot, XAxis, YAxis, minMax, Area } from '@semcore/ui/d3-chart';
import { scaleLinear } from 'd3-scale';
import { curveCardinal } from 'd3-shape';
function formatDate(value, options) {
return new Intl.DateTimeFormat('en', options).format(value);
}
const Demo = () => {
const MARGIN = 40;
const width = 500;
const height = 300;
const xScale = scaleLinear()
.range([MARGIN, width - MARGIN])
.domain(minMax(data, 'time'));
const yScale = scaleLinear()
.range([height - MARGIN, MARGIN])
.domain([0, 10]);
return (
<Plot data={data} scale={[xScale, yScale]} width={width} height={height}>
<YAxis>
<YAxis.Ticks />
<YAxis.Grid />
</YAxis>
<XAxis>
<XAxis.Ticks ticks={data.map((d) => +d.time)}>
{({ value }) => ({
children: formatDate(value, {
month: 'short',
day: 'numeric',
}),
})}
</XAxis.Ticks>
</XAxis>
<Area x='time' y='line' curve={curveCardinal}>
<Area.Dots display />
</Area>
</Plot>
);
};
const date = new Date();
const data = Array(10)
.fill({})
.map((d, i) => {
return {
time: new Date(date.setDate(date.getDate() + 5)),
line: Math.random() * 10,
};
});
export default Demo;
Edge cases
- If a part of the chart has no data – use a dashed line to draw the period.
- If the data has only one value – display it as a dot.
- Two consecutively known values will automatically be displayed as the
Area
component.
import React from 'react';
import { Plot, XAxis, YAxis, minMax, HoverLine, Area } from '@semcore/ui/d3-chart';
import { scaleLinear } from 'd3-scale';
import { Flex } from '@semcore/ui/flex-box';
import { Text } from '@semcore/ui/typography';
const Demo = () => {
const MARGIN = 40;
const width = 500;
const height = 300;
const xScale = scaleLinear()
.range([MARGIN, width - MARGIN])
.domain(minMax(data, 'x'));
const yScale = scaleLinear()
.range([height - MARGIN, MARGIN])
.domain([0, 10]);
return (
<Plot data={data} scale={[xScale, yScale]} width={width} height={height}>
<YAxis>
<YAxis.Ticks />
<YAxis.Grid />
</YAxis>
<XAxis>
<XAxis.Ticks />
</XAxis>
<HoverLine.Tooltip x='x' wMin={100}>
{({ xIndex }) => {
return {
children: (
<>
<HoverLine.Tooltip.Title>{data[xIndex].x}</HoverLine.Tooltip.Title>
<Flex justifyContent='space-between'>
<HoverLine.Tooltip.Dot mr={4}>Line</HoverLine.Tooltip.Dot>
<Text bold>{data[xIndex].y ?? 'n/a'}</Text>
</Flex>
</>
),
};
}}
</HoverLine.Tooltip>
<Area x='x' y='y'>
<Area.Null />
<Area.Dots />
</Area>
</Plot>
);
};
const data = [
{ x: 0, y: 1 },
{ x: 1, y: 4 },
{ x: 2, y: null },
{ x: 3, y: null },
{ x: 4, y: 1 },
{ x: 5, y: null },
];
export default Demo;
Custom line
import React from 'react';
import { scaleLinear } from 'd3-scale';
import { curveCardinal } from 'd3-shape';
import { Area, minMax, Plot, XAxis, YAxis } from '@semcore/ui/d3-chart';
const customLineStyles = { strokeWidth: 4, stroke: 'pink' };
const Demo = () => {
const MARGIN = 40;
const width = 500;
const height = 300;
const xScale = scaleLinear()
.range([MARGIN, width - MARGIN])
.domain(minMax(data, 'x'));
const yScale = scaleLinear()
.range([height - MARGIN, MARGIN])
.domain([0, 10]);
return (
<Plot data={data} scale={[xScale, yScale]} width={width} height={height}>
<YAxis>
<YAxis.Ticks />
</YAxis>
<XAxis>
<XAxis.Ticks />
</XAxis>
<Area x='x' y='y' curve={curveCardinal}>
<Area.Line style={customLineStyles} />
</Area>
</Plot>
);
};
const data = Array(10)
.fill({})
.map((d, i) => {
return {
x: i,
y: Math.random() * 10,
};
});
export default Demo;
Interpolation
If exact values of specific point is not available, you can pass interpolateValue
and value will be automatically interpolated.
WARNING
🚨 Interpolation doesn't works with StackedArea
.
import React from 'react';
import { Plot, XAxis, YAxis, minMax, Area, interpolateValue } from '@semcore/ui/d3-chart';
import { scaleLinear } from 'd3-scale';
import { curveCardinal } from 'd3-shape';
function formatDate(value, options) {
return new Intl.DateTimeFormat('en', options).format(value);
}
const Demo = () => {
const MARGIN = 40;
const width = 500;
const height = 300;
const xScale = scaleLinear()
.range([MARGIN, width - MARGIN])
.domain(minMax(data, 'time'));
const yScale = scaleLinear()
.range([height - MARGIN, MARGIN])
.domain([0, 10]);
return (
<Plot data={data} scale={[xScale, yScale]} width={width} height={height}>
<YAxis>
<YAxis.Ticks />
<YAxis.Grid />
</YAxis>
<XAxis>
<XAxis.Ticks ticks={data.map((d) => +d.time)}>
{({ value }) => ({
children: formatDate(value, {
month: 'short',
day: 'numeric',
}),
})}
</XAxis.Ticks>
</XAxis>
<Area x='time' y='line1' curve={curveCardinal}>
<Area.Dots display />
</Area>
<Area x='time' y='line2' curve={curveCardinal}>
<Area.Dots display />
</Area>
</Plot>
);
};
const data = [
{
time: new Date(Date.now() + 5 * 60 * 60 * 1000),
line1: 5,
line2: 3,
},
{
time: new Date(Date.now() + 10 * 60 * 60 * 1000),
line1: 8,
line2: interpolateValue,
},
{
time: new Date(Date.now() + 15 * 60 * 60 * 1000),
line1: 4,
line2: 8,
},
{
time: new Date(Date.now() + 20 * 60 * 60 * 1000),
line1: 5,
line2: interpolateValue,
},
{
time: new Date(Date.now() + 25 * 60 * 60 * 1000),
line1: 5,
line2: interpolateValue,
},
{
time: new Date(Date.now() + 30 * 60 * 60 * 1000),
line1: 3,
line2: 1,
},
];
export default Demo;
Legend and pattern fill
To make data available without relying only on colors (for example, for different kinds of colorblind and high-contrast modes), use the patterns
property.
Note that for ChartLegend patterns
property works only with default shape={'Checkbox'}
.
import React from 'react';
import {
Plot,
XAxis,
YAxis,
minMax,
Area,
interpolateValue,
ChartLegend,
makeDataHintsContainer,
} from '@semcore/ui/d3-chart';
import { scaleLinear } from 'd3-scale';
import { curveCardinal } from 'd3-shape';
function formatDate(value, options) {
return new Intl.DateTimeFormat('en', options).format(value);
}
const dataHints = makeDataHintsContainer();
const Demo = () => {
const MARGIN = 40;
const width = 500;
const height = 300;
const xScale = scaleLinear()
.range([MARGIN, width - MARGIN])
.domain(minMax(data, 'time'));
const yScale = scaleLinear()
.range([height - MARGIN, MARGIN])
.domain([0, 10]);
const [legendItems, setLegendItems] = React.useState(
Object.keys(data[0])
.filter((name) => name !== 'time')
.map((item, index) => {
return {
id: item,
label: `Line (${item})`,
checked: true,
color: `chart-palette-order-${index + 1}`,
};
}),
);
const handleChangeVisible = React.useCallback((id: string, isVisible: boolean) => {
setLegendItems((prevItems) => {
return prevItems.map((item) => {
if (item.id === id) {
item.checked = isVisible;
}
return item;
});
});
}, []);
return (
<>
<ChartLegend
dataHints={dataHints}
items={legendItems}
onChangeVisibleItem={handleChangeVisible}
patterns
aria-label={'Legend for the area chart'}
/>
<Plot
data={data}
scale={[xScale, yScale]}
width={width}
height={height}
dataHints={dataHints}
patterns={true}
>
<YAxis>
<YAxis.Ticks />
<YAxis.Grid />
</YAxis>
<XAxis>
<XAxis.Ticks ticks={data.map((d) => +d.time)}>
{({ value }) => ({
children: formatDate(value, {
month: 'short',
day: 'numeric',
}),
})}
</XAxis.Ticks>
</XAxis>
{legendItems.map((item) => {
return (
item.checked && (
<Area key={item.id} x='time' y={item.id} curve={curveCardinal} color={item.color}>
<Area.Dots display />
</Area>
)
);
})}
</Plot>
</>
);
};
const data = [
{
time: new Date(Date.now() + 5 * 60 * 60 * 1000),
line1: 5,
line2: 3,
},
{
time: new Date(Date.now() + 10 * 60 * 60 * 1000),
line1: 8,
line2: interpolateValue,
},
{
time: new Date(Date.now() + 15 * 60 * 60 * 1000),
line1: 4,
line2: 8,
},
{
time: new Date(Date.now() + 20 * 60 * 60 * 1000),
line1: 5,
line2: interpolateValue,
},
{
time: new Date(Date.now() + 25 * 60 * 60 * 1000),
line1: 5,
line2: interpolateValue,
},
{
time: new Date(Date.now() + 30 * 60 * 60 * 1000),
line1: 3,
line2: 1,
},
];
export default Demo;