2022-09-28 03:24:42 +00:00
|
|
|
import { useMemo, useRef } from 'react'
|
2022-09-29 04:14:34 +00:00
|
|
|
import { range } from 'lodash'
|
2022-09-28 08:00:39 +00:00
|
|
|
import { scaleLinear } from 'd3-scale'
|
2022-09-28 03:24:42 +00:00
|
|
|
|
2022-09-29 04:14:34 +00:00
|
|
|
import { formatLargeNumber } from 'common/util/format'
|
2022-09-28 03:24:42 +00:00
|
|
|
import { getDpmOutcomeProbabilities } from 'common/calculate-dpm'
|
|
|
|
import { NumericContract } from 'common/contract'
|
|
|
|
import { NUMERIC_GRAPH_COLOR } from 'common/numeric-constants'
|
|
|
|
import { useIsMobile } from 'web/hooks/use-is-mobile'
|
2022-09-29 19:51:38 +00:00
|
|
|
import { TooltipProps, MARGIN_X, MARGIN_Y, formatPct } from '../helpers'
|
|
|
|
import { DistributionPoint, DistributionChart } from '../generic-charts'
|
2022-09-28 03:24:42 +00:00
|
|
|
import { useElementWidth } from 'web/hooks/use-element-width'
|
|
|
|
|
|
|
|
const getNumericChartData = (contract: NumericContract) => {
|
|
|
|
const { totalShares, bucketCount, min, max } = contract
|
|
|
|
const step = (max - min) / bucketCount
|
|
|
|
const bucketProbs = getDpmOutcomeProbabilities(totalShares)
|
2022-09-29 04:14:34 +00:00
|
|
|
return range(bucketCount).map((i) => ({
|
|
|
|
x: min + step * (i + 0.5),
|
|
|
|
y: bucketProbs[`${i}`],
|
|
|
|
}))
|
|
|
|
}
|
|
|
|
|
2022-09-30 23:16:04 +00:00
|
|
|
const NumericChartTooltip = (
|
|
|
|
props: TooltipProps<number, DistributionPoint>
|
|
|
|
) => {
|
|
|
|
const { data, mouseX, xScale } = props
|
|
|
|
const x = xScale.invert(mouseX)
|
2022-09-29 04:14:34 +00:00
|
|
|
return (
|
2022-09-30 05:45:51 +00:00
|
|
|
<>
|
|
|
|
<span className="text-semibold">{formatLargeNumber(x)}</span>
|
2022-09-30 23:16:04 +00:00
|
|
|
<span className="text-greyscale-6">{formatPct(data.y, 2)}</span>
|
2022-09-30 05:45:51 +00:00
|
|
|
</>
|
2022-09-28 03:24:42 +00:00
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
export const NumericContractChart = (props: {
|
|
|
|
contract: NumericContract
|
|
|
|
height?: number
|
2022-09-30 03:18:33 +00:00
|
|
|
onMouseOver?: (p: DistributionPoint | undefined) => void
|
2022-09-28 03:24:42 +00:00
|
|
|
}) => {
|
2022-09-30 03:18:33 +00:00
|
|
|
const { contract, onMouseOver } = props
|
2022-09-29 04:14:34 +00:00
|
|
|
const { min, max } = contract
|
2022-09-28 03:24:42 +00:00
|
|
|
const data = useMemo(() => getNumericChartData(contract), [contract])
|
|
|
|
const isMobile = useIsMobile(800)
|
|
|
|
const containerRef = useRef<HTMLDivElement>(null)
|
|
|
|
const width = useElementWidth(containerRef) ?? 0
|
|
|
|
const height = props.height ?? (isMobile ? 150 : 250)
|
2022-09-29 04:14:34 +00:00
|
|
|
const maxY = Math.max(...data.map((d) => d.y))
|
|
|
|
const xScale = scaleLinear([min, max], [0, width - MARGIN_X])
|
2022-09-28 03:24:42 +00:00
|
|
|
const yScale = scaleLinear([0, maxY], [height - MARGIN_Y, 0])
|
|
|
|
return (
|
|
|
|
<div ref={containerRef}>
|
2022-09-28 21:20:28 +00:00
|
|
|
{width > 0 && (
|
2022-09-29 19:51:38 +00:00
|
|
|
<DistributionChart
|
2022-09-28 03:24:42 +00:00
|
|
|
w={width}
|
|
|
|
h={height}
|
|
|
|
xScale={xScale}
|
|
|
|
yScale={yScale}
|
|
|
|
data={data}
|
|
|
|
color={NUMERIC_GRAPH_COLOR}
|
2022-09-30 03:18:33 +00:00
|
|
|
onMouseOver={onMouseOver}
|
2022-09-29 04:14:34 +00:00
|
|
|
Tooltip={NumericChartTooltip}
|
2022-09-28 03:24:42 +00:00
|
|
|
/>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
)
|
|
|
|
}
|