2022-09-29 04:43:04 +00:00
|
|
|
import {
|
|
|
|
ReactNode,
|
|
|
|
SVGProps,
|
|
|
|
memo,
|
|
|
|
useRef,
|
|
|
|
useEffect,
|
|
|
|
useMemo,
|
|
|
|
useState,
|
|
|
|
} from 'react'
|
|
|
|
import { pointer, select } from 'd3-selection'
|
2022-09-29 19:51:38 +00:00
|
|
|
import { Axis, AxisScale } from 'd3-axis'
|
2022-09-28 08:00:39 +00:00
|
|
|
import { brushX, D3BrushEvent } from 'd3-brush'
|
|
|
|
import { area, line, curveStepAfter, CurveFactory } from 'd3-shape'
|
2022-09-28 03:24:42 +00:00
|
|
|
import { nanoid } from 'nanoid'
|
2022-09-29 04:14:34 +00:00
|
|
|
import dayjs from 'dayjs'
|
2022-09-28 03:24:42 +00:00
|
|
|
import clsx from 'clsx'
|
|
|
|
|
|
|
|
import { Contract } from 'common/contract'
|
|
|
|
|
2022-09-30 23:16:04 +00:00
|
|
|
export type Point<X, Y, T = unknown> = { x: X; y: Y; obj?: T }
|
|
|
|
|
|
|
|
export interface ContinuousScale<T> extends AxisScale<T> {
|
|
|
|
invert(n: number): T
|
|
|
|
}
|
|
|
|
|
2022-09-29 19:51:38 +00:00
|
|
|
export type XScale<P> = P extends Point<infer X, infer _> ? AxisScale<X> : never
|
|
|
|
export type YScale<P> = P extends Point<infer _, infer Y> ? AxisScale<Y> : never
|
|
|
|
|
2022-09-28 03:24:42 +00:00
|
|
|
export const MARGIN = { top: 20, right: 10, bottom: 20, left: 40 }
|
|
|
|
export const MARGIN_X = MARGIN.right + MARGIN.left
|
|
|
|
export const MARGIN_Y = MARGIN.top + MARGIN.bottom
|
2022-09-30 05:45:31 +00:00
|
|
|
const MARGIN_STYLE = `${MARGIN.top}px ${MARGIN.right}px ${MARGIN.bottom}px ${MARGIN.left}px`
|
|
|
|
const MARGIN_XFORM = `translate(${MARGIN.left}, ${MARGIN.top})`
|
2022-09-28 03:24:42 +00:00
|
|
|
|
|
|
|
export const XAxis = <X,>(props: { w: number; h: number; axis: Axis<X> }) => {
|
|
|
|
const { h, axis } = props
|
|
|
|
const axisRef = useRef<SVGGElement>(null)
|
|
|
|
useEffect(() => {
|
|
|
|
if (axisRef.current != null) {
|
|
|
|
select(axisRef.current)
|
|
|
|
.transition()
|
|
|
|
.duration(250)
|
|
|
|
.call(axis)
|
|
|
|
.select('.domain')
|
|
|
|
.attr('stroke-width', 0)
|
|
|
|
}
|
|
|
|
}, [h, axis])
|
|
|
|
return <g ref={axisRef} transform={`translate(0, ${h})`} />
|
|
|
|
}
|
|
|
|
|
|
|
|
export const YAxis = <Y,>(props: { w: number; h: number; axis: Axis<Y> }) => {
|
|
|
|
const { w, h, axis } = props
|
|
|
|
const axisRef = useRef<SVGGElement>(null)
|
|
|
|
useEffect(() => {
|
|
|
|
if (axisRef.current != null) {
|
|
|
|
select(axisRef.current)
|
|
|
|
.transition()
|
|
|
|
.duration(250)
|
|
|
|
.call(axis)
|
|
|
|
.call((g) =>
|
|
|
|
g.selectAll('.tick line').attr('x2', w).attr('stroke-opacity', 0.1)
|
|
|
|
)
|
|
|
|
.select('.domain')
|
|
|
|
.attr('stroke-width', 0)
|
|
|
|
}
|
|
|
|
}, [w, h, axis])
|
|
|
|
return <g ref={axisRef} />
|
|
|
|
}
|
|
|
|
|
|
|
|
const LinePathInternal = <P,>(
|
|
|
|
props: {
|
|
|
|
data: P[]
|
|
|
|
px: number | ((p: P) => number)
|
|
|
|
py: number | ((p: P) => number)
|
|
|
|
curve?: CurveFactory
|
|
|
|
} & SVGProps<SVGPathElement>
|
|
|
|
) => {
|
|
|
|
const { data, px, py, curve, ...rest } = props
|
|
|
|
const d3Line = line<P>(px, py).curve(curve ?? curveStepAfter)
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
|
|
|
|
return <path {...rest} fill="none" d={d3Line(data)!} />
|
|
|
|
}
|
|
|
|
export const LinePath = memo(LinePathInternal) as typeof LinePathInternal
|
|
|
|
|
|
|
|
const AreaPathInternal = <P,>(
|
|
|
|
props: {
|
|
|
|
data: P[]
|
|
|
|
px: number | ((p: P) => number)
|
|
|
|
py0: number | ((p: P) => number)
|
|
|
|
py1: number | ((p: P) => number)
|
|
|
|
curve?: CurveFactory
|
|
|
|
} & SVGProps<SVGPathElement>
|
|
|
|
) => {
|
|
|
|
const { data, px, py0, py1, curve, ...rest } = props
|
|
|
|
const d3Area = area<P>(px, py0, py1).curve(curve ?? curveStepAfter)
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
|
|
|
|
return <path {...rest} d={d3Area(data)!} />
|
|
|
|
}
|
|
|
|
export const AreaPath = memo(AreaPathInternal) as typeof AreaPathInternal
|
|
|
|
|
|
|
|
export const AreaWithTopStroke = <P,>(props: {
|
|
|
|
color: string
|
|
|
|
data: P[]
|
|
|
|
px: number | ((p: P) => number)
|
|
|
|
py0: number | ((p: P) => number)
|
|
|
|
py1: number | ((p: P) => number)
|
|
|
|
curve?: CurveFactory
|
|
|
|
}) => {
|
|
|
|
const { color, data, px, py0, py1, curve } = props
|
|
|
|
return (
|
|
|
|
<g>
|
|
|
|
<AreaPath
|
|
|
|
data={data}
|
|
|
|
px={px}
|
|
|
|
py0={py0}
|
|
|
|
py1={py1}
|
|
|
|
curve={curve}
|
|
|
|
fill={color}
|
2022-09-30 07:03:31 +00:00
|
|
|
opacity={0.2}
|
2022-09-28 03:24:42 +00:00
|
|
|
/>
|
|
|
|
<LinePath data={data} px={px} py={py1} curve={curve} stroke={color} />
|
|
|
|
</g>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2022-09-30 23:16:04 +00:00
|
|
|
export const SVGChart = <X, TT>(props: {
|
2022-09-28 03:24:42 +00:00
|
|
|
children: ReactNode
|
|
|
|
w: number
|
|
|
|
h: number
|
|
|
|
xAxis: Axis<X>
|
2022-09-29 19:51:38 +00:00
|
|
|
yAxis: Axis<number>
|
2022-09-28 03:24:42 +00:00
|
|
|
onSelect?: (ev: D3BrushEvent<any>) => void
|
2022-09-30 23:16:04 +00:00
|
|
|
onMouseOver?: (mouseX: number, mouseY: number) => TT | undefined
|
|
|
|
Tooltip?: TooltipComponent<X, TT>
|
2022-09-28 03:24:42 +00:00
|
|
|
}) => {
|
2022-09-29 04:43:04 +00:00
|
|
|
const { children, w, h, xAxis, yAxis, onMouseOver, onSelect, Tooltip } = props
|
2022-09-30 23:16:04 +00:00
|
|
|
const [mouse, setMouse] = useState<{ x: number; y: number; data: TT }>()
|
2022-09-28 03:24:42 +00:00
|
|
|
const overlayRef = useRef<SVGGElement>(null)
|
|
|
|
const innerW = w - MARGIN_X
|
|
|
|
const innerH = h - MARGIN_Y
|
|
|
|
const clipPathId = useMemo(() => nanoid(), [])
|
|
|
|
|
|
|
|
const justSelected = useRef(false)
|
|
|
|
useEffect(() => {
|
|
|
|
if (onSelect != null && overlayRef.current) {
|
|
|
|
const brush = brushX().extent([
|
|
|
|
[0, 0],
|
|
|
|
[innerW, innerH],
|
|
|
|
])
|
|
|
|
brush.on('end', (ev) => {
|
|
|
|
// when we clear the brush after a selection, that would normally cause
|
|
|
|
// another 'end' event, so we have to suppress it with this flag
|
|
|
|
if (!justSelected.current) {
|
|
|
|
justSelected.current = true
|
|
|
|
onSelect(ev)
|
2022-09-30 23:16:04 +00:00
|
|
|
setMouse(undefined)
|
2022-09-28 03:24:42 +00:00
|
|
|
if (overlayRef.current) {
|
|
|
|
select(overlayRef.current).call(brush.clear)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
justSelected.current = false
|
|
|
|
}
|
|
|
|
})
|
2022-09-28 07:58:51 +00:00
|
|
|
// mqp: shape-rendering null overrides the default d3-brush shape-rendering
|
|
|
|
// of `crisp-edges`, which seems to cause graphical glitches on Chrome
|
|
|
|
// (i.e. the bug where the area fill flickers white)
|
|
|
|
select(overlayRef.current)
|
|
|
|
.call(brush)
|
|
|
|
.select('.selection')
|
|
|
|
.attr('shape-rendering', 'null')
|
2022-09-28 03:24:42 +00:00
|
|
|
}
|
|
|
|
}, [innerW, innerH, onSelect])
|
|
|
|
|
2022-09-29 04:43:04 +00:00
|
|
|
const onPointerMove = (ev: React.PointerEvent) => {
|
|
|
|
if (ev.pointerType === 'mouse' && onMouseOver) {
|
2022-09-30 23:16:04 +00:00
|
|
|
const [x, y] = pointer(ev)
|
|
|
|
const data = onMouseOver(x, y)
|
|
|
|
if (data !== undefined) {
|
|
|
|
setMouse({ x, y, data })
|
2022-09-29 04:43:04 +00:00
|
|
|
} else {
|
2022-09-30 23:16:04 +00:00
|
|
|
setMouse(undefined)
|
2022-09-29 04:43:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const onPointerLeave = () => {
|
2022-09-30 23:16:04 +00:00
|
|
|
setMouse(undefined)
|
2022-09-29 04:43:04 +00:00
|
|
|
}
|
|
|
|
|
2022-09-28 03:24:42 +00:00
|
|
|
return (
|
2022-09-29 04:43:04 +00:00
|
|
|
<div className="relative">
|
2022-09-30 23:16:04 +00:00
|
|
|
{mouse && Tooltip && (
|
|
|
|
<TooltipContainer
|
|
|
|
pos={getTooltipPosition(mouse.x, mouse.y, innerW, innerH)}
|
|
|
|
>
|
|
|
|
<Tooltip
|
|
|
|
xScale={xAxis.scale()}
|
|
|
|
mouseX={mouse.x}
|
|
|
|
mouseY={mouse.y}
|
|
|
|
data={mouse.data}
|
|
|
|
/>
|
2022-09-29 04:43:04 +00:00
|
|
|
</TooltipContainer>
|
|
|
|
)}
|
2022-09-30 05:45:31 +00:00
|
|
|
<svg width={w} height={h} viewBox={`0 0 ${w} ${h}`}>
|
2022-09-29 04:43:04 +00:00
|
|
|
<clipPath id={clipPathId}>
|
|
|
|
<rect x={0} y={0} width={innerW} height={innerH} />
|
|
|
|
</clipPath>
|
2022-09-30 05:45:31 +00:00
|
|
|
<g transform={MARGIN_XFORM}>
|
2022-09-29 04:43:04 +00:00
|
|
|
<XAxis axis={xAxis} w={innerW} h={innerH} />
|
|
|
|
<YAxis axis={yAxis} w={innerW} h={innerH} />
|
|
|
|
<g clipPath={`url(#${clipPathId})`}>{children}</g>
|
|
|
|
<g
|
|
|
|
ref={overlayRef}
|
|
|
|
x="0"
|
|
|
|
y="0"
|
|
|
|
width={innerW}
|
|
|
|
height={innerH}
|
|
|
|
fill="none"
|
|
|
|
pointerEvents="all"
|
|
|
|
onPointerEnter={onPointerMove}
|
|
|
|
onPointerMove={onPointerMove}
|
|
|
|
onPointerLeave={onPointerLeave}
|
|
|
|
/>
|
|
|
|
</g>
|
|
|
|
</svg>
|
|
|
|
</div>
|
2022-09-28 03:24:42 +00:00
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2022-09-30 05:45:31 +00:00
|
|
|
export type TooltipPosition = {
|
|
|
|
top?: number
|
|
|
|
right?: number
|
|
|
|
bottom?: number
|
|
|
|
left?: number
|
|
|
|
}
|
|
|
|
|
|
|
|
export const getTooltipPosition = (
|
|
|
|
mouseX: number,
|
|
|
|
mouseY: number,
|
|
|
|
w: number,
|
|
|
|
h: number
|
|
|
|
) => {
|
|
|
|
const result: TooltipPosition = {}
|
|
|
|
if (mouseX <= (3 * w) / 4) {
|
|
|
|
result.left = mouseX + 10 // in the left three quarters
|
|
|
|
} else {
|
|
|
|
result.right = w - mouseX + 10 // in the right quarter
|
|
|
|
}
|
|
|
|
if (mouseY <= h / 4) {
|
|
|
|
result.top = mouseY + 10 // in the top quarter
|
|
|
|
} else {
|
|
|
|
result.bottom = h - mouseY + 10 // in the bottom three quarters
|
|
|
|
}
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
2022-09-30 23:16:04 +00:00
|
|
|
export type TooltipProps<X, T> = {
|
|
|
|
mouseX: number
|
|
|
|
mouseY: number
|
|
|
|
xScale: ContinuousScale<X>
|
|
|
|
data: T
|
|
|
|
}
|
|
|
|
export type TooltipComponent<X, T> = React.ComponentType<TooltipProps<X, T>>
|
2022-09-30 05:45:31 +00:00
|
|
|
export const TooltipContainer = (props: {
|
|
|
|
pos: TooltipPosition
|
|
|
|
className?: string
|
|
|
|
children: React.ReactNode
|
|
|
|
}) => {
|
|
|
|
const { pos, className, children } = props
|
2022-09-28 03:24:42 +00:00
|
|
|
return (
|
|
|
|
<div
|
|
|
|
className={clsx(
|
|
|
|
className,
|
2022-09-30 07:03:31 +00:00
|
|
|
'pointer-events-none absolute z-10 whitespace-pre rounded border border-gray-200 bg-white/80 p-2 px-4 py-2 text-xs sm:text-sm'
|
2022-09-28 03:24:42 +00:00
|
|
|
)}
|
2022-09-30 05:45:31 +00:00
|
|
|
style={{ margin: MARGIN_STYLE, ...pos }}
|
2022-09-28 03:24:42 +00:00
|
|
|
>
|
|
|
|
{children}
|
|
|
|
</div>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
export const getDateRange = (contract: Contract) => {
|
|
|
|
const { createdTime, closeTime, resolutionTime } = contract
|
|
|
|
const isClosed = !!closeTime && Date.now() > closeTime
|
|
|
|
const endDate = resolutionTime ?? (isClosed ? closeTime : null)
|
2022-09-30 04:35:20 +00:00
|
|
|
return [createdTime, endDate ?? null] as const
|
2022-09-28 03:24:42 +00:00
|
|
|
}
|
2022-09-28 04:18:22 +00:00
|
|
|
|
|
|
|
export const getRightmostVisibleDate = (
|
2022-09-30 04:35:20 +00:00
|
|
|
contractEnd: number | null | undefined,
|
|
|
|
lastActivity: number | null | undefined,
|
|
|
|
now: number
|
2022-09-28 04:18:22 +00:00
|
|
|
) => {
|
|
|
|
if (contractEnd != null) {
|
|
|
|
return contractEnd
|
|
|
|
} else if (lastActivity != null) {
|
|
|
|
// client-DB clock divergence may cause last activity to be later than now
|
2022-09-30 04:35:20 +00:00
|
|
|
return Math.max(lastActivity, now)
|
2022-09-28 04:18:22 +00:00
|
|
|
} else {
|
|
|
|
return now
|
|
|
|
}
|
|
|
|
}
|
2022-09-29 04:14:34 +00:00
|
|
|
|
|
|
|
export const formatPct = (n: number, digits?: number) => {
|
|
|
|
return `${(n * 100).toFixed(digits ?? 0)}%`
|
|
|
|
}
|
|
|
|
|
|
|
|
export const formatDate = (
|
|
|
|
date: Date,
|
|
|
|
opts: { includeYear: boolean; includeHour: boolean; includeMinute: boolean }
|
|
|
|
) => {
|
|
|
|
const { includeYear, includeHour, includeMinute } = opts
|
|
|
|
const d = dayjs(date)
|
|
|
|
const now = Date.now()
|
|
|
|
if (
|
|
|
|
d.add(1, 'minute').isAfter(now) &&
|
|
|
|
d.subtract(1, 'minute').isBefore(now)
|
|
|
|
) {
|
|
|
|
return 'Now'
|
|
|
|
} else {
|
|
|
|
const dayName = d.isSame(now, 'day')
|
|
|
|
? 'Today'
|
|
|
|
: d.add(1, 'day').isSame(now, 'day')
|
|
|
|
? 'Yesterday'
|
|
|
|
: null
|
|
|
|
let format = dayName ? `[${dayName}]` : 'MMM D'
|
|
|
|
if (includeMinute) {
|
|
|
|
format += ', h:mma'
|
|
|
|
} else if (includeHour) {
|
|
|
|
format += ', ha'
|
|
|
|
} else if (includeYear) {
|
|
|
|
format += ', YYYY'
|
|
|
|
}
|
|
|
|
return d.format(format)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export const formatDateInRange = (d: Date, start: Date, end: Date) => {
|
|
|
|
const opts = {
|
|
|
|
includeYear: !dayjs(start).isSame(end, 'year'),
|
|
|
|
includeHour: dayjs(start).add(8, 'day').isAfter(end),
|
|
|
|
includeMinute: dayjs(end).diff(start, 'hours') < 2,
|
|
|
|
}
|
|
|
|
return formatDate(d, opts)
|
|
|
|
}
|