2022-09-28 03:24:42 +00:00
|
|
|
import { useCallback, useMemo, useState } from 'react'
|
2022-09-28 08:00:39 +00:00
|
|
|
import { bisector } from 'd3-array'
|
|
|
|
import { axisBottom, axisLeft } from 'd3-axis'
|
|
|
|
import { D3BrushEvent } from 'd3-brush'
|
|
|
|
import { ScaleTime, ScaleContinuousNumeric } from 'd3-scale'
|
|
|
|
import { pointer } from 'd3-selection'
|
2022-09-28 03:24:42 +00:00
|
|
|
import {
|
|
|
|
curveLinear,
|
|
|
|
curveStepAfter,
|
|
|
|
stack,
|
|
|
|
stackOrderReverse,
|
|
|
|
SeriesPoint,
|
2022-09-28 08:00:39 +00:00
|
|
|
} from 'd3-shape'
|
2022-09-29 04:14:34 +00:00
|
|
|
import { range } from 'lodash'
|
2022-09-28 03:24:42 +00:00
|
|
|
|
|
|
|
import {
|
|
|
|
SVGChart,
|
|
|
|
AreaPath,
|
|
|
|
AreaWithTopStroke,
|
2022-09-29 04:14:34 +00:00
|
|
|
TooltipContent,
|
|
|
|
TooltipContainer,
|
2022-09-28 03:24:42 +00:00
|
|
|
TooltipPosition,
|
2022-09-29 04:14:34 +00:00
|
|
|
formatPct,
|
2022-09-28 03:24:42 +00:00
|
|
|
} from './helpers'
|
|
|
|
import { useEvent } from 'web/hooks/use-event'
|
|
|
|
|
2022-09-29 04:14:34 +00:00
|
|
|
export type MultiPoint<T = never> = { x: Date; y: number[]; datum?: T }
|
|
|
|
export type HistoryPoint<T = never> = { x: Date; y: number; datum?: T }
|
|
|
|
export type DistributionPoint<T = never> = { x: number; y: number; datum?: T }
|
2022-09-28 03:24:42 +00:00
|
|
|
|
2022-09-29 04:14:34 +00:00
|
|
|
type PositionValue<P> = TooltipPosition & { p: P }
|
2022-09-28 03:24:42 +00:00
|
|
|
|
|
|
|
const getTickValues = (min: number, max: number, n: number) => {
|
|
|
|
const step = (max - min) / (n - 1)
|
|
|
|
return [min, ...range(1, n - 1).map((i) => min + step * i), max]
|
|
|
|
}
|
|
|
|
|
2022-09-29 04:14:34 +00:00
|
|
|
export const SingleValueDistributionChart = <T,>(props: {
|
|
|
|
data: DistributionPoint<T>[]
|
2022-09-28 03:24:42 +00:00
|
|
|
w: number
|
|
|
|
h: number
|
|
|
|
color: string
|
|
|
|
xScale: ScaleContinuousNumeric<number, number>
|
|
|
|
yScale: ScaleContinuousNumeric<number, number>
|
2022-09-29 04:14:34 +00:00
|
|
|
Tooltip?: TooltipContent<SingleValueDistributionTooltipProps<T>>
|
2022-09-28 03:24:42 +00:00
|
|
|
}) => {
|
2022-09-29 04:14:34 +00:00
|
|
|
const { color, data, yScale, w, h, Tooltip } = props
|
2022-09-28 03:24:42 +00:00
|
|
|
|
|
|
|
const [viewXScale, setViewXScale] =
|
|
|
|
useState<ScaleContinuousNumeric<number, number>>()
|
|
|
|
const [mouseState, setMouseState] =
|
2022-09-29 04:14:34 +00:00
|
|
|
useState<PositionValue<DistributionPoint<T>>>()
|
2022-09-28 03:24:42 +00:00
|
|
|
const xScale = viewXScale ?? props.xScale
|
|
|
|
|
2022-09-29 04:14:34 +00:00
|
|
|
const px = useCallback((p: DistributionPoint<T>) => xScale(p.x), [xScale])
|
2022-09-28 03:24:42 +00:00
|
|
|
const py0 = yScale(yScale.domain()[0])
|
2022-09-29 04:14:34 +00:00
|
|
|
const py1 = useCallback((p: DistributionPoint<T>) => yScale(p.y), [yScale])
|
|
|
|
const xBisector = bisector((p: DistributionPoint<T>) => p.x)
|
2022-09-28 03:24:42 +00:00
|
|
|
|
2022-09-29 04:14:34 +00:00
|
|
|
const { xAxis, yAxis } = useMemo(() => {
|
2022-09-28 03:24:42 +00:00
|
|
|
const xAxis = axisBottom<number>(xScale).ticks(w / 100)
|
2022-09-29 04:14:34 +00:00
|
|
|
const yAxis = axisLeft<number>(yScale).tickFormat((n) => formatPct(n, 2))
|
|
|
|
return { xAxis, yAxis }
|
2022-09-28 03:24:42 +00:00
|
|
|
}, [w, xScale, yScale])
|
|
|
|
|
2022-09-29 04:14:34 +00:00
|
|
|
const onSelect = useEvent((ev: D3BrushEvent<DistributionPoint<T>>) => {
|
2022-09-28 03:24:42 +00:00
|
|
|
if (ev.selection) {
|
|
|
|
const [mouseX0, mouseX1] = ev.selection as [number, number]
|
|
|
|
setViewXScale(() =>
|
|
|
|
xScale.copy().domain([xScale.invert(mouseX0), xScale.invert(mouseX1)])
|
|
|
|
)
|
|
|
|
setMouseState(undefined)
|
|
|
|
} else {
|
|
|
|
setViewXScale(undefined)
|
|
|
|
setMouseState(undefined)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
const onMouseOver = useEvent((ev: React.PointerEvent) => {
|
|
|
|
if (ev.pointerType === 'mouse') {
|
|
|
|
const [mouseX, mouseY] = pointer(ev)
|
|
|
|
const queryX = xScale.invert(mouseX)
|
2022-09-28 07:56:43 +00:00
|
|
|
const item = data[xBisector.left(data, queryX) - 1]
|
|
|
|
if (item == null) {
|
|
|
|
// this can happen if you are on the very left or right edge of the chart,
|
|
|
|
// so your queryX is out of bounds
|
|
|
|
return
|
|
|
|
}
|
2022-09-29 04:14:34 +00:00
|
|
|
const p = { x: queryX, y: item.y, datum: item.datum }
|
|
|
|
setMouseState({ top: mouseY - 10, left: mouseX + 60, p })
|
2022-09-28 03:24:42 +00:00
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
const onMouseLeave = useEvent(() => {
|
|
|
|
setMouseState(undefined)
|
|
|
|
})
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div className="relative">
|
2022-09-29 04:14:34 +00:00
|
|
|
{mouseState && Tooltip && (
|
|
|
|
<TooltipContainer className="text-sm" {...mouseState}>
|
|
|
|
<Tooltip xScale={xScale} {...mouseState.p} />
|
|
|
|
</TooltipContainer>
|
2022-09-28 03:24:42 +00:00
|
|
|
)}
|
|
|
|
<SVGChart
|
|
|
|
w={w}
|
|
|
|
h={h}
|
|
|
|
xAxis={xAxis}
|
|
|
|
yAxis={yAxis}
|
|
|
|
onSelect={onSelect}
|
|
|
|
onMouseOver={onMouseOver}
|
|
|
|
onMouseLeave={onMouseLeave}
|
|
|
|
>
|
|
|
|
<AreaWithTopStroke
|
|
|
|
color={color}
|
|
|
|
data={data}
|
|
|
|
px={px}
|
|
|
|
py0={py0}
|
|
|
|
py1={py1}
|
|
|
|
curve={curveLinear}
|
|
|
|
/>
|
|
|
|
</SVGChart>
|
|
|
|
</div>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2022-09-29 04:14:34 +00:00
|
|
|
export type SingleValueDistributionTooltipProps<T = unknown> =
|
|
|
|
DistributionPoint<T> & {
|
|
|
|
xScale: React.ComponentProps<
|
|
|
|
typeof SingleValueDistributionChart<T>
|
|
|
|
>['xScale']
|
|
|
|
}
|
|
|
|
|
|
|
|
export const MultiValueHistoryChart = <T,>(props: {
|
|
|
|
data: MultiPoint<T>[]
|
2022-09-28 03:24:42 +00:00
|
|
|
w: number
|
|
|
|
h: number
|
|
|
|
colors: readonly string[]
|
|
|
|
xScale: ScaleTime<number, number>
|
|
|
|
yScale: ScaleContinuousNumeric<number, number>
|
2022-09-29 04:14:34 +00:00
|
|
|
Tooltip?: TooltipContent<MultiValueHistoryTooltipProps<T>>
|
2022-09-28 03:24:42 +00:00
|
|
|
pct?: boolean
|
|
|
|
}) => {
|
2022-09-29 04:14:34 +00:00
|
|
|
const { colors, data, yScale, w, h, Tooltip, pct } = props
|
2022-09-28 03:24:42 +00:00
|
|
|
|
|
|
|
const [viewXScale, setViewXScale] = useState<ScaleTime<number, number>>()
|
2022-09-29 04:14:34 +00:00
|
|
|
const [mouseState, setMouseState] = useState<PositionValue<MultiPoint<T>>>()
|
2022-09-28 03:24:42 +00:00
|
|
|
const xScale = viewXScale ?? props.xScale
|
|
|
|
|
2022-09-29 04:14:34 +00:00
|
|
|
type SP = SeriesPoint<MultiPoint<T>>
|
|
|
|
const px = useCallback((p: SP) => xScale(p.data.x), [xScale])
|
2022-09-28 03:24:42 +00:00
|
|
|
const py0 = useCallback((p: SP) => yScale(p[0]), [yScale])
|
|
|
|
const py1 = useCallback((p: SP) => yScale(p[1]), [yScale])
|
2022-09-29 04:14:34 +00:00
|
|
|
const xBisector = bisector((p: MultiPoint<T>) => p.x)
|
2022-09-28 03:24:42 +00:00
|
|
|
|
2022-09-29 04:14:34 +00:00
|
|
|
const { xAxis, yAxis } = useMemo(() => {
|
2022-09-28 03:24:42 +00:00
|
|
|
const [min, max] = yScale.domain()
|
|
|
|
const pctTickValues = getTickValues(min, max, h < 200 ? 3 : 5)
|
|
|
|
const xAxis = axisBottom<Date>(xScale).ticks(w / 100)
|
|
|
|
const yAxis = pct
|
2022-09-29 04:14:34 +00:00
|
|
|
? axisLeft<number>(yScale).tickValues(pctTickValues).tickFormat(formatPct)
|
2022-09-28 03:24:42 +00:00
|
|
|
: axisLeft<number>(yScale)
|
2022-09-29 04:14:34 +00:00
|
|
|
return { xAxis, yAxis }
|
2022-09-28 03:24:42 +00:00
|
|
|
}, [w, h, pct, xScale, yScale])
|
|
|
|
|
|
|
|
const series = useMemo(() => {
|
2022-09-29 04:14:34 +00:00
|
|
|
const d3Stack = stack<MultiPoint<T>, number>()
|
|
|
|
.keys(range(0, Math.max(...data.map(({ y }) => y.length))))
|
|
|
|
.value(({ y }, o) => y[o])
|
2022-09-28 03:24:42 +00:00
|
|
|
.order(stackOrderReverse)
|
|
|
|
return d3Stack(data)
|
2022-09-29 04:14:34 +00:00
|
|
|
}, [data])
|
2022-09-28 03:24:42 +00:00
|
|
|
|
2022-09-29 04:14:34 +00:00
|
|
|
const onSelect = useEvent((ev: D3BrushEvent<MultiPoint<T>>) => {
|
2022-09-28 03:24:42 +00:00
|
|
|
if (ev.selection) {
|
|
|
|
const [mouseX0, mouseX1] = ev.selection as [number, number]
|
|
|
|
setViewXScale(() =>
|
|
|
|
xScale.copy().domain([xScale.invert(mouseX0), xScale.invert(mouseX1)])
|
|
|
|
)
|
|
|
|
setMouseState(undefined)
|
|
|
|
} else {
|
|
|
|
setViewXScale(undefined)
|
|
|
|
setMouseState(undefined)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
const onMouseOver = useEvent((ev: React.PointerEvent) => {
|
|
|
|
if (ev.pointerType === 'mouse') {
|
|
|
|
const [mouseX, mouseY] = pointer(ev)
|
|
|
|
const queryX = xScale.invert(mouseX)
|
2022-09-28 07:56:43 +00:00
|
|
|
const item = data[xBisector.left(data, queryX) - 1]
|
|
|
|
if (item == null) {
|
|
|
|
// this can happen if you are on the very left or right edge of the chart,
|
|
|
|
// so your queryX is out of bounds
|
|
|
|
return
|
|
|
|
}
|
2022-09-29 04:14:34 +00:00
|
|
|
const p = { x: queryX, y: item.y, datum: item.datum }
|
|
|
|
setMouseState({ top: mouseY - 10, left: mouseX + 60, p })
|
2022-09-28 03:24:42 +00:00
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
const onMouseLeave = useEvent(() => {
|
|
|
|
setMouseState(undefined)
|
|
|
|
})
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div className="relative">
|
2022-09-29 04:14:34 +00:00
|
|
|
{mouseState && Tooltip && (
|
|
|
|
<TooltipContainer top={mouseState.top} left={mouseState.left}>
|
|
|
|
<Tooltip xScale={xScale} {...mouseState.p} />
|
|
|
|
</TooltipContainer>
|
2022-09-28 03:24:42 +00:00
|
|
|
)}
|
|
|
|
<SVGChart
|
|
|
|
w={w}
|
|
|
|
h={h}
|
|
|
|
xAxis={xAxis}
|
|
|
|
yAxis={yAxis}
|
|
|
|
onSelect={onSelect}
|
|
|
|
onMouseOver={onMouseOver}
|
|
|
|
onMouseLeave={onMouseLeave}
|
|
|
|
>
|
|
|
|
{series.map((s, i) => (
|
|
|
|
<AreaPath
|
|
|
|
key={i}
|
|
|
|
data={s}
|
|
|
|
px={px}
|
|
|
|
py0={py0}
|
|
|
|
py1={py1}
|
|
|
|
curve={curveStepAfter}
|
|
|
|
fill={colors[i]}
|
|
|
|
/>
|
|
|
|
))}
|
|
|
|
</SVGChart>
|
|
|
|
</div>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2022-09-29 04:14:34 +00:00
|
|
|
export type MultiValueHistoryTooltipProps<T = unknown> = MultiPoint<T> & {
|
|
|
|
xScale: React.ComponentProps<typeof MultiValueHistoryChart<T>>['xScale']
|
|
|
|
}
|
|
|
|
|
|
|
|
export const SingleValueHistoryChart = <T,>(props: {
|
|
|
|
data: HistoryPoint<T>[]
|
2022-09-28 03:24:42 +00:00
|
|
|
w: number
|
|
|
|
h: number
|
|
|
|
color: string
|
|
|
|
xScale: ScaleTime<number, number>
|
|
|
|
yScale: ScaleContinuousNumeric<number, number>
|
2022-09-29 04:14:34 +00:00
|
|
|
Tooltip?: TooltipContent<SingleValueHistoryTooltipProps<T>>
|
2022-09-28 03:24:42 +00:00
|
|
|
pct?: boolean
|
|
|
|
}) => {
|
2022-09-29 04:14:34 +00:00
|
|
|
const { color, data, pct, yScale, w, h, Tooltip } = props
|
2022-09-28 03:24:42 +00:00
|
|
|
|
|
|
|
const [viewXScale, setViewXScale] = useState<ScaleTime<number, number>>()
|
2022-09-29 04:14:34 +00:00
|
|
|
const [mouseState, setMouseState] = useState<PositionValue<HistoryPoint<T>>>()
|
2022-09-28 03:24:42 +00:00
|
|
|
const xScale = viewXScale ?? props.xScale
|
|
|
|
|
2022-09-29 04:14:34 +00:00
|
|
|
const px = useCallback((p: HistoryPoint<T>) => xScale(p.x), [xScale])
|
2022-09-28 03:24:42 +00:00
|
|
|
const py0 = yScale(yScale.domain()[0])
|
2022-09-29 04:14:34 +00:00
|
|
|
const py1 = useCallback((p: HistoryPoint<T>) => yScale(p.y), [yScale])
|
|
|
|
const xBisector = bisector((p: HistoryPoint<T>) => p.x)
|
2022-09-28 03:24:42 +00:00
|
|
|
|
2022-09-29 04:14:34 +00:00
|
|
|
const { xAxis, yAxis } = useMemo(() => {
|
2022-09-28 03:24:42 +00:00
|
|
|
const [min, max] = yScale.domain()
|
|
|
|
const pctTickValues = getTickValues(min, max, h < 200 ? 3 : 5)
|
|
|
|
const xAxis = axisBottom<Date>(xScale).ticks(w / 100)
|
|
|
|
const yAxis = pct
|
2022-09-29 04:14:34 +00:00
|
|
|
? axisLeft<number>(yScale).tickValues(pctTickValues).tickFormat(formatPct)
|
2022-09-28 03:24:42 +00:00
|
|
|
: axisLeft<number>(yScale)
|
2022-09-29 04:14:34 +00:00
|
|
|
return { xAxis, yAxis }
|
2022-09-28 03:24:42 +00:00
|
|
|
}, [w, h, pct, xScale, yScale])
|
|
|
|
|
2022-09-29 04:14:34 +00:00
|
|
|
const onSelect = useEvent((ev: D3BrushEvent<HistoryPoint<T>>) => {
|
2022-09-28 03:24:42 +00:00
|
|
|
if (ev.selection) {
|
|
|
|
const [mouseX0, mouseX1] = ev.selection as [number, number]
|
|
|
|
setViewXScale(() =>
|
|
|
|
xScale.copy().domain([xScale.invert(mouseX0), xScale.invert(mouseX1)])
|
|
|
|
)
|
|
|
|
setMouseState(undefined)
|
|
|
|
} else {
|
|
|
|
setViewXScale(undefined)
|
|
|
|
setMouseState(undefined)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
const onMouseOver = useEvent((ev: React.PointerEvent) => {
|
|
|
|
if (ev.pointerType === 'mouse') {
|
|
|
|
const [mouseX, mouseY] = pointer(ev)
|
|
|
|
const queryX = xScale.invert(mouseX)
|
2022-09-28 07:56:43 +00:00
|
|
|
const item = data[xBisector.left(data, queryX) - 1]
|
|
|
|
if (item == null) {
|
|
|
|
// this can happen if you are on the very left or right edge of the chart,
|
|
|
|
// so your queryX is out of bounds
|
|
|
|
return
|
|
|
|
}
|
2022-09-29 04:14:34 +00:00
|
|
|
const p = { x: queryX, y: item.y, datum: item.datum }
|
|
|
|
setMouseState({ top: mouseY - 10, left: mouseX + 60, p })
|
2022-09-28 03:24:42 +00:00
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
const onMouseLeave = useEvent(() => {
|
|
|
|
setMouseState(undefined)
|
|
|
|
})
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div className="relative">
|
2022-09-29 04:14:34 +00:00
|
|
|
{mouseState && Tooltip && (
|
|
|
|
<TooltipContainer top={mouseState.top} left={mouseState.left}>
|
|
|
|
<Tooltip xScale={xScale} {...mouseState.p} />
|
|
|
|
</TooltipContainer>
|
2022-09-28 03:24:42 +00:00
|
|
|
)}
|
|
|
|
<SVGChart
|
|
|
|
w={w}
|
|
|
|
h={h}
|
|
|
|
xAxis={xAxis}
|
|
|
|
yAxis={yAxis}
|
|
|
|
onSelect={onSelect}
|
|
|
|
onMouseOver={onMouseOver}
|
|
|
|
onMouseLeave={onMouseLeave}
|
|
|
|
>
|
|
|
|
<AreaWithTopStroke
|
|
|
|
color={color}
|
|
|
|
data={data}
|
|
|
|
px={px}
|
|
|
|
py0={py0}
|
|
|
|
py1={py1}
|
|
|
|
curve={curveStepAfter}
|
|
|
|
/>
|
|
|
|
</SVGChart>
|
|
|
|
</div>
|
|
|
|
)
|
|
|
|
}
|
2022-09-29 04:14:34 +00:00
|
|
|
|
|
|
|
export type SingleValueHistoryTooltipProps<T = unknown> = HistoryPoint<T> & {
|
|
|
|
xScale: React.ComponentProps<typeof SingleValueHistoryChart<T>>['xScale']
|
|
|
|
}
|