bf8dca25b2
* Make curve configurable on generic charts * Extract SizedContainer helper component * Use new charts for stats page * Move analytics charts component * Fix up start date logic for graphs excluding data
257 lines
6.9 KiB
TypeScript
257 lines
6.9 KiB
TypeScript
import { useCallback, useMemo, useState } from 'react'
|
|
import { bisector } from 'd3-array'
|
|
import { axisBottom, axisLeft } from 'd3-axis'
|
|
import { D3BrushEvent } from 'd3-brush'
|
|
import { ScaleTime, ScaleContinuousNumeric } from 'd3-scale'
|
|
import {
|
|
CurveFactory,
|
|
SeriesPoint,
|
|
curveLinear,
|
|
stack,
|
|
stackOrderReverse,
|
|
} from 'd3-shape'
|
|
import { range } from 'lodash'
|
|
|
|
import {
|
|
ContinuousScale,
|
|
SVGChart,
|
|
AreaPath,
|
|
AreaWithTopStroke,
|
|
Point,
|
|
TooltipComponent,
|
|
formatPct,
|
|
} from './helpers'
|
|
import { useEvent } from 'web/hooks/use-event'
|
|
|
|
export type MultiPoint<T = unknown> = Point<Date, number[], T>
|
|
export type HistoryPoint<T = unknown> = Point<Date, number, T>
|
|
export type DistributionPoint<T = unknown> = Point<number, number, T>
|
|
|
|
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]
|
|
}
|
|
|
|
const betAtPointSelector = <X, Y, P extends Point<X, Y>>(
|
|
data: P[],
|
|
xScale: ContinuousScale<X>
|
|
) => {
|
|
const bisect = bisector((p: P) => p.x)
|
|
return (posX: number) => {
|
|
const x = xScale.invert(posX)
|
|
const item = data[bisect.left(data, x) - 1]
|
|
const result = item ? { ...item, x: posX } : undefined
|
|
return result
|
|
}
|
|
}
|
|
|
|
export const DistributionChart = <P extends DistributionPoint>(props: {
|
|
data: P[]
|
|
w: number
|
|
h: number
|
|
color: string
|
|
xScale: ScaleContinuousNumeric<number, number>
|
|
yScale: ScaleContinuousNumeric<number, number>
|
|
curve?: CurveFactory
|
|
onMouseOver?: (p: P | undefined) => void
|
|
Tooltip?: TooltipComponent<number, P>
|
|
}) => {
|
|
const { color, data, yScale, w, h, curve, Tooltip } = props
|
|
|
|
const [viewXScale, setViewXScale] =
|
|
useState<ScaleContinuousNumeric<number, number>>()
|
|
const xScale = viewXScale ?? props.xScale
|
|
|
|
const px = useCallback((p: P) => xScale(p.x), [xScale])
|
|
const py0 = yScale(yScale.domain()[0])
|
|
const py1 = useCallback((p: P) => yScale(p.y), [yScale])
|
|
|
|
const { xAxis, yAxis } = useMemo(() => {
|
|
const xAxis = axisBottom<number>(xScale).ticks(w / 100)
|
|
const yAxis = axisLeft<number>(yScale).tickFormat((n) => formatPct(n, 2))
|
|
return { xAxis, yAxis }
|
|
}, [w, xScale, yScale])
|
|
|
|
const onMouseOver = useEvent(betAtPointSelector(data, xScale))
|
|
|
|
const onSelect = useEvent((ev: D3BrushEvent<P>) => {
|
|
if (ev.selection) {
|
|
const [mouseX0, mouseX1] = ev.selection as [number, number]
|
|
setViewXScale(() =>
|
|
xScale.copy().domain([xScale.invert(mouseX0), xScale.invert(mouseX1)])
|
|
)
|
|
} else {
|
|
setViewXScale(undefined)
|
|
}
|
|
})
|
|
|
|
return (
|
|
<SVGChart
|
|
w={w}
|
|
h={h}
|
|
xAxis={xAxis}
|
|
yAxis={yAxis}
|
|
onSelect={onSelect}
|
|
onMouseOver={onMouseOver}
|
|
Tooltip={Tooltip}
|
|
>
|
|
<AreaWithTopStroke
|
|
color={color}
|
|
data={data}
|
|
px={px}
|
|
py0={py0}
|
|
py1={py1}
|
|
curve={curve ?? curveLinear}
|
|
/>
|
|
</SVGChart>
|
|
)
|
|
}
|
|
|
|
export const MultiValueHistoryChart = <P extends MultiPoint>(props: {
|
|
data: P[]
|
|
w: number
|
|
h: number
|
|
colors: readonly string[]
|
|
xScale: ScaleTime<number, number>
|
|
yScale: ScaleContinuousNumeric<number, number>
|
|
curve?: CurveFactory
|
|
onMouseOver?: (p: P | undefined) => void
|
|
Tooltip?: TooltipComponent<Date, P>
|
|
pct?: boolean
|
|
}) => {
|
|
const { colors, data, yScale, w, h, curve, Tooltip, pct } = props
|
|
|
|
const [viewXScale, setViewXScale] = useState<ScaleTime<number, number>>()
|
|
const xScale = viewXScale ?? props.xScale
|
|
|
|
type SP = SeriesPoint<P>
|
|
const px = useCallback((p: SP) => xScale(p.data.x), [xScale])
|
|
const py0 = useCallback((p: SP) => yScale(p[0]), [yScale])
|
|
const py1 = useCallback((p: SP) => yScale(p[1]), [yScale])
|
|
|
|
const { xAxis, yAxis } = useMemo(() => {
|
|
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
|
|
? axisLeft<number>(yScale)
|
|
.tickValues(pctTickValues)
|
|
.tickFormat((n) => formatPct(n))
|
|
: axisLeft<number>(yScale)
|
|
return { xAxis, yAxis }
|
|
}, [w, h, pct, xScale, yScale])
|
|
|
|
const series = useMemo(() => {
|
|
const d3Stack = stack<P, number>()
|
|
.keys(range(0, Math.max(...data.map(({ y }) => y.length))))
|
|
.value(({ y }, o) => y[o])
|
|
.order(stackOrderReverse)
|
|
return d3Stack(data)
|
|
}, [data])
|
|
|
|
const onMouseOver = useEvent(betAtPointSelector(data, xScale))
|
|
|
|
const onSelect = useEvent((ev: D3BrushEvent<P>) => {
|
|
if (ev.selection) {
|
|
const [mouseX0, mouseX1] = ev.selection as [number, number]
|
|
setViewXScale(() =>
|
|
xScale.copy().domain([xScale.invert(mouseX0), xScale.invert(mouseX1)])
|
|
)
|
|
} else {
|
|
setViewXScale(undefined)
|
|
}
|
|
})
|
|
|
|
return (
|
|
<SVGChart
|
|
w={w}
|
|
h={h}
|
|
xAxis={xAxis}
|
|
yAxis={yAxis}
|
|
onSelect={onSelect}
|
|
onMouseOver={onMouseOver}
|
|
Tooltip={Tooltip}
|
|
>
|
|
{series.map((s, i) => (
|
|
<AreaPath
|
|
key={i}
|
|
data={s}
|
|
px={px}
|
|
py0={py0}
|
|
py1={py1}
|
|
curve={curve ?? curveLinear}
|
|
fill={colors[i]}
|
|
/>
|
|
))}
|
|
</SVGChart>
|
|
)
|
|
}
|
|
|
|
export const SingleValueHistoryChart = <P extends HistoryPoint>(props: {
|
|
data: P[]
|
|
w: number
|
|
h: number
|
|
color: string
|
|
xScale: ScaleTime<number, number>
|
|
yScale: ScaleContinuousNumeric<number, number>
|
|
curve?: CurveFactory
|
|
onMouseOver?: (p: P | undefined) => void
|
|
Tooltip?: TooltipComponent<Date, P>
|
|
pct?: boolean
|
|
}) => {
|
|
const { color, data, yScale, w, h, curve, Tooltip, pct } = props
|
|
|
|
const [viewXScale, setViewXScale] = useState<ScaleTime<number, number>>()
|
|
const xScale = viewXScale ?? props.xScale
|
|
|
|
const px = useCallback((p: P) => xScale(p.x), [xScale])
|
|
const py0 = yScale(yScale.domain()[0])
|
|
const py1 = useCallback((p: P) => yScale(p.y), [yScale])
|
|
|
|
const { xAxis, yAxis } = useMemo(() => {
|
|
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
|
|
? axisLeft<number>(yScale)
|
|
.tickValues(pctTickValues)
|
|
.tickFormat((n) => formatPct(n))
|
|
: axisLeft<number>(yScale)
|
|
return { xAxis, yAxis }
|
|
}, [w, h, pct, xScale, yScale])
|
|
|
|
const onMouseOver = useEvent(betAtPointSelector(data, xScale))
|
|
|
|
const onSelect = useEvent((ev: D3BrushEvent<P>) => {
|
|
if (ev.selection) {
|
|
const [mouseX0, mouseX1] = ev.selection as [number, number]
|
|
setViewXScale(() =>
|
|
xScale.copy().domain([xScale.invert(mouseX0), xScale.invert(mouseX1)])
|
|
)
|
|
} else {
|
|
setViewXScale(undefined)
|
|
}
|
|
})
|
|
|
|
return (
|
|
<SVGChart
|
|
w={w}
|
|
h={h}
|
|
xAxis={xAxis}
|
|
yAxis={yAxis}
|
|
onSelect={onSelect}
|
|
onMouseOver={onMouseOver}
|
|
Tooltip={Tooltip}
|
|
>
|
|
<AreaWithTopStroke
|
|
color={color}
|
|
data={data}
|
|
px={px}
|
|
py0={py0}
|
|
py1={py1}
|
|
curve={curve ?? curveLinear}
|
|
/>
|
|
</SVGChart>
|
|
)
|
|
}
|