Skip to content

Fan Chart API

Plot the forecast and its uncertainty in one chart: history, a dashed median, and confidence bands that widen with the horizon - see the Fan Chart demo.

Import

ts
import "@michi-vz/wc/fan-chart";
// <michi-vz-fan-chart> is now defined
ts
import { mountFanChart } from "@michi-vz/core";

const chart = mountFanChart(el, props);
ts
import { forecastFan } from "@michi-vz/insights/forecast";

const item = forecastFan(history, { horizon: 4, levels: [0.5, 0.8], level: 0.95 }, "Revenue");
const chart = mountFanChart(el, { dataSet: [item], xAxisDataType: "date_annual" });

Props

PropTypeDefaultDescription
dataSet*FanDataItem[]Array of forecast series, each a median line plus its nested confidence bands
titlestringOptional chart title rendered above the plot
yAxisDomain[number, number]Fixed [min, max] for the y-axis; when omitted the domain auto-fits the line plus the widest band extents
xAxisDataType"date_annual" | "date_monthly" | "number""number"How x values are parsed and formatted: yearly dates, monthly dates, or plain numbers
xAxisFormat(d: number | string) => stringFormats an x tick value into its display label
yAxisFormat(d: number | string) => stringFormats a y tick value into its display label
ticksnumber5Approximate number of axis ticks to generate
tickValuesArray<number | Date>Explicit tick values, overriding the generated ones
curve"curveBumpX" | "curveLinear" | "curveMonotoneX"Line interpolation: curveLinear, curveMonotoneX, or curveBumpX
fillOpacitynumber0.18band fill opacity for the widest band (narrower bands scale up from here).
forecastZonebooleanshade the forecast region (from the last solid point to the end). Default true.
showDataPointsbooleanfalseDraw a marker at each median data point (default false)
tooltipFormatter(item: FanDataItem, lastPoint: DataPoint | null) => stringCustom HTML for the tooltip (DOMPurify-sanitized; may include `<a href>` links).
onHighlightItem(labels: string[]) => voidCalled when the hovered/highlighted label(s) change
Common props — shared by every chart (14)
PropTypeDefaultDescription
widthnumber1000Chart width in pixels
heightnumber500Chart height in pixels
marginMargin{ top: 50, right: 50, bottom: 50, left: 60 }Inner margins (top/right/bottom/left, in px) reserved for axes, titles, and labels
colorsstring[]Categorical palette for series/labels without an explicit colour or colorsMapping entry
colorsMappingRecord<string, string>Explicit label -> colour map; takes precedence over the palette and per-item colours
highlightItemsstring[]Labels to emphasise; all other marks dim
disabledItemsstring[]Labels to hide and exclude from scales/stacks
renderer"svg" | "canvas""svg"Render as inline SVG (default) or to a canvas (faster for large datasets); getContext() is identical either way
localestringBCP-47 locale used for number and date formatting
skipColorMappingDispatchbooleanfalseExternal-CSS mode: unmapped labels resolve to transparent and onColorMappingGenerated is not emitted, so mark colours come from your CSS via the data-label-safe contract
enableTransitionsbooleantrueAnimate updates with CSS transitions (default true)
onColorMappingGenerated(mapping: Record<string, string>) => voidCalled with the resolved label -> colour map after the chart assigns colours
onChartDataProcessed(context: ChartContext) => voidCalled with the renderer-agnostic ChartContext whenever the data is (re)processed
onDataWarning(warnings: DataWarning[]) => voidCalled with any non-fatal data warnings (duplicate labels, non-finite values, gaps, ...)

Events

The web component dispatches these bubbling CustomEvents (the engine exposes the same via the on* callbacks in the table above):

EventDetailFires when
michi-vz:highlightstring[]hover highlight changes
michi-vz:colormappingRecord<string, string>a color mapping is generated
michi-vz:dataprocessedChartContextdata is (re)processed
michi-vz:datawarningDataWarning[]input warnings are detected

getContext()

mountFanChart(el, props).getContext() returns a renderer-agnostic FanChartContext (per-series history/forecast counts, band levels, final uncertainty, plus a deterministic natural-language summary + an a11y table). See LLM context.

Source

Props are typed as FanChartProps in @michi-vz/core.

Free and open source. MIT licensed.