Stacked 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';
import { curveCardinal } from 'd3-shape';
const formatDate = (type: 'axis' | 'tooltip') => (value) => {
const options =
type === 'axis'
? {
month: 'short' as const,
day: 'numeric' as const,
}
: {
year: 'numeric' as const,
month: 'long' as const,
day: 'numeric' as const,
};
return new Intl.DateTimeFormat('en', options).format(value);
};
const Demo = () => {
return (
<Chart.Area
data={data}
plotWidth={500}
plotHeight={200}
groupKey={'time'}
tooltipValueFormatter={formatDate('tooltip')}
axisXValueFormatter={formatDate('axis')}
stacked={true}
curve={curveCardinal}
aria-label={'Stacked area chart'}
/>
);
};
const date = new Date();
const data = [...Array(5).keys()].map((d, i) => ({
time: new Date(date.setDate(date.getDate() + 5)),
stack1: Math.random() * 5,
stack2: Math.random() * 5,
stack3: Math.random() * 5,
}));
export default Demo;
Stacked area
If you need to display a part-to-whole ratio – use the <StackedArea/>
and <StackedArea.Area/>
components.
import React from 'react';
import { Plot, XAxis, YAxis, minMax, StackedArea, HoverLine } from '@semcore/ui/d3-chart';
import { scaleLinear } from 'd3-scale';
import { Flex, Box } from '@semcore/ui/flex-box';
import { Text } from '@semcore/ui/typography';
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, 15]);
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>
<HoverLine.Tooltip x='time' wMin={100}>
{({ xIndex }) => {
return {
children: (
<>
<HoverLine.Tooltip.Title>
{formatDate(data[xIndex].time, {
year: 'numeric',
month: 'long',
day: 'numeric',
})}
</HoverLine.Tooltip.Title>
<Flex justifyContent='space-between'>
<HoverLine.Tooltip.Dot mr={4}>Stack 1</HoverLine.Tooltip.Dot>
<Text bold>{data[xIndex].stack1}</Text>
</Flex>
<Flex mt={2} justifyContent='space-between'>
<HoverLine.Tooltip.Dot mr={4}>Stack 2</HoverLine.Tooltip.Dot>
<Text bold>{data[xIndex].stack2}</Text>
</Flex>
<Flex mt={2} justifyContent='space-between'>
<HoverLine.Tooltip.Dot mr={4}>Stack 3</HoverLine.Tooltip.Dot>
<Text bold>{data[xIndex].stack3}</Text>
</Flex>
<Flex mt={2} justifyContent='space-between'>
<Box mr={4}>Total</Box>
<Text bold>
{data[xIndex].stack1 + data[xIndex].stack2 + data[xIndex].stack3}
</Text>
</Flex>
</>
),
};
}}
</HoverLine.Tooltip>
<StackedArea x='time'>
<StackedArea.Area y='stack1' curve={curveCardinal}>
<StackedArea.Area.Dots />
</StackedArea.Area>
<StackedArea.Area y='stack2' curve={curveCardinal}>
<StackedArea.Area.Dots />
</StackedArea.Area>
<StackedArea.Area y='stack3' curve={curveCardinal}>
<StackedArea.Area.Dots />
</StackedArea.Area>
</StackedArea>
</Plot>
);
};
const date = new Date();
const data = [...Array(5).keys()].map((d, i) => ({
time: new Date(date.setDate(date.getDate() + 5)),
stack1: Math.random() * 5,
stack2: Math.random() * 5,
stack3: Math.random() * 5,
}));
export default Demo;
Edge cases
- If a part of the chart has no data – use a dashed line to draw that period.
- If the data has only one value – display it as a dot.
- Two consecutively known values will automatically be displayed as the
StackedArea
component.
import React from 'react';
import { Plot, XAxis, YAxis, minMax, StackedArea, HoverLine } 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, 'time'));
const yScale = scaleLinear()
.range([height - MARGIN, MARGIN])
.domain([0, 15]);
return (
<Plot data={data} scale={[xScale, yScale]} width={width} height={height} patterns>
<YAxis>
<YAxis.Ticks />
<YAxis.Grid />
</YAxis>
<XAxis>
<XAxis.Ticks ticks={data.map((d) => +d.time)} />
</XAxis>
<HoverLine.Tooltip x='time' wMin={100}>
{({ xIndex }) => {
return {
children: (
<>
<HoverLine.Tooltip.Title>{data[xIndex].time}</HoverLine.Tooltip.Title>
<Flex justifyContent='space-between'>
<HoverLine.Tooltip.Dot mr={4}>Stack 1</HoverLine.Tooltip.Dot>
<Text bold>{data[xIndex].stack1 ?? 'n/a'}</Text>
</Flex>
<Flex mt={2} justifyContent='space-between'>
<HoverLine.Tooltip.Dot mr={4}>Stack 2</HoverLine.Tooltip.Dot>
<Text bold>{data[xIndex].stack2 ?? 'n/a'}</Text>
</Flex>
<Flex mt={2} justifyContent='space-between'>
<HoverLine.Tooltip.Dot mr={4}>Stack 3</HoverLine.Tooltip.Dot>
<Text bold>{data[xIndex].stack3 ?? 'n/a'}</Text>
</Flex>
</>
),
};
}}
</HoverLine.Tooltip>
<StackedArea x='time'>
<StackedArea.Area y='stack1'>
<StackedArea.Area.Null />
<StackedArea.Area.Dots />
</StackedArea.Area>
<StackedArea.Area y='stack2'>
<StackedArea.Area.Null />
<StackedArea.Area.Dots />
</StackedArea.Area>
<StackedArea.Area y='stack3'>
<StackedArea.Area.Null />
<StackedArea.Area.Dots />
</StackedArea.Area>
</StackedArea>
</Plot>
);
};
const data = [
{ time: 0, stack1: 1, stack2: 4, stack3: 3 },
{ time: 1, stack1: 2, stack2: 3, stack3: 4 },
{ time: 2, stack1: 1, stack2: 4, stack3: 5 },
{ time: 3, stack1: null, stack2: null, stack3: null },
{ time: 4, stack1: null, stack2: null, stack3: null },
{ time: 5, stack1: 3, stack2: 4, stack3: 3 },
{ time: 6, stack1: null, stack2: null, stack3: null },
{ time: 7, stack1: 2, stack2: 5, stack3: 3 },
{ time: 8, stack1: 2, stack2: 6, stack3: 5 },
{ time: 9, stack1: 5, stack2: 5, stack3: 3 },
];
export default Demo;
Legend and pattern fill
Note that for ChartLegend patterns
property works only with default shape={'Checkbox'}
.
import React from 'react';
import {
Plot,
XAxis,
YAxis,
minMax,
StackedArea,
HoverLine,
makeDataHintsContainer,
LegendItem,
ChartLegend,
} from '@semcore/ui/d3-chart';
import { scaleLinear } from 'd3-scale';
import { Flex, Box } from '@semcore/ui/flex-box';
import { Text } from '@semcore/ui/typography';
import { curveCardinal } from 'd3-shape';
import resolveColor from '@semcore/utils/lib/color';
function formatDate(value, options) {
return new Intl.DateTimeFormat('en', options).format(value);
}
const lineColors = {
1: resolveColor('blue-300'),
2: resolveColor('green-200'),
3: resolveColor('orange-400'),
};
const dataHints = makeDataHintsContainer();
const getDegaultLegendItems = () => {
return Object.keys(data[0])
.filter((name) => name !== 'time')
.map((item) => {
return {
id: item,
label: `Stack ${item}`,
checked: true,
color: lineColors[item],
};
});
};
const Demo = () => {
const [legendItems, setLegendItems] = React.useState(getDegaultLegendItems);
const handleChangeVisible = React.useCallback((id: string, isVisible: boolean) => {
setLegendItems((prevItems) => {
const newItems = prevItems.map((item) => {
if (item.id === id) {
item.checked = isVisible;
}
return item;
});
return newItems;
});
}, []);
const MARGIN = 28;
const width = 500;
const height = 260;
const xScale = scaleLinear()
.range([MARGIN, width - MARGIN])
.domain(minMax(data, 'time'));
const yScale = scaleLinear()
.range([height - MARGIN, MARGIN])
.domain([0, 15]);
return (
<>
<ChartLegend
dataHints={dataHints}
items={legendItems}
shape={'Checkbox'}
patterns
aria-label={'Legend for the stacked area chart'}
onChangeVisibleItem={handleChangeVisible}
/>
<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>
<HoverLine.Tooltip x='time' wMin={100}>
{({ xIndex }) => {
return {
children: (
<>
<HoverLine.Tooltip.Title>
{formatDate(data[xIndex].time, {
year: 'numeric',
month: 'long',
day: 'numeric',
})}
</HoverLine.Tooltip.Title>
{legendItems.map((item, index) => {
const itemIndex = index + 1;
return (
<Flex key={item.id} justifyContent='space-between'>
<HoverLine.Tooltip.Dot mr={4} color={lineColors[itemIndex]}>
{item.label}
</HoverLine.Tooltip.Dot>
<Text bold>{data[xIndex][itemIndex]}</Text>
</Flex>
);
})}
<Flex mt={2} justifyContent='space-between'>
<Box mr={4}>Total</Box>
<Text bold>{data[xIndex][1] + data[xIndex][2] + data[xIndex][3]}</Text>
</Flex>
</>
),
};
}}
</HoverLine.Tooltip>
<StackedArea x='time'>
{legendItems.map((item, index) => {
const itemIndex = String(index + 1);
return (
item.checked && (
<StackedArea.Area
key={item.id}
y={itemIndex}
fill={`chart-palette-order-${itemIndex}`}
color={lineColors[itemIndex]}
curve={curveCardinal}
>
<StackedArea.Area.Dots />
</StackedArea.Area>
)
);
})}
</StackedArea>
</Plot>
</>
);
};
const date = new Date();
const data = [...Array(5).keys()].map((d, i) => ({
time: new Date(date.setDate(date.getDate() + 5)),
1: Math.random() * 5,
2: Math.random() * 5,
3: Math.random() * 5,
}));
export default Demo;