123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175 |
- import type {ReactText} from 'react';
- import {Fragment} from 'react';
- import styled from '@emotion/styled';
- import Duration from 'sentry/components/duration';
- import FileSize from 'sentry/components/fileSize';
- import LoadingIndicator from 'sentry/components/loadingIndicator';
- import {PercentChange} from 'sentry/components/percentChange';
- import {Tooltip} from 'sentry/components/tooltip';
- import {defined} from 'sentry/utils';
- import {
- type CountUnit,
- CurrencyUnit,
- DurationUnit,
- type PercentageUnit,
- type PercentChangeUnit,
- RateUnit,
- SizeUnit,
- } from 'sentry/utils/discover/fields';
- import {
- formatAbbreviatedNumber,
- formatPercentage,
- formatRate,
- } from 'sentry/utils/formatters';
- import {Block} from 'sentry/views/starfish/views/spanSummaryPage/block';
- type Unit =
- | DurationUnit.MILLISECOND
- | SizeUnit.BYTE
- | RateUnit
- | CountUnit
- | PercentageUnit
- | PercentChangeUnit
- | CurrencyUnit;
- interface Props {
- title: string;
- unit: Unit;
- value: ReactText | undefined;
- align?: 'left' | 'right';
- isLoading?: boolean;
- tooltip?: React.ReactNode;
- }
- export function MetricReadout(props: Props) {
- return (
- <Block title={props.title} alignment={props.align}>
- <ReadoutContent {...props} />
- </Block>
- );
- }
- function ReadoutContent({unit, value, tooltip, align = 'right', isLoading}: Props) {
- if (isLoading) {
- return (
- <LoadingContainer align={align}>
- <LoadingIndicator mini />
- </LoadingContainer>
- );
- }
- if (!defined(value)) {
- return <Fragment>--</Fragment>;
- }
- let renderedValue: React.ReactNode;
- if (isARateUnit(unit)) {
- renderedValue = (
- <NumberContainer align={align}>
- {formatRate(typeof value === 'string' ? parseFloat(value) : value, unit, {
- minimumValue: MINIMUM_RATE_VALUE,
- })}
- </NumberContainer>
- );
- }
- if (unit === DurationUnit.MILLISECOND) {
- // TODO: Implement other durations
- renderedValue = (
- <NumberContainer align={align}>
- <Duration
- seconds={typeof value === 'string' ? parseFloat(value) : value / 1000}
- fixedDigits={2}
- abbreviation
- />
- </NumberContainer>
- );
- }
- if (unit === SizeUnit.BYTE) {
- // TODO: Implement other sizes
- renderedValue = (
- <NumberContainer align={align}>
- <FileSize bytes={typeof value === 'string' ? parseInt(value, 10) : value} />
- </NumberContainer>
- );
- }
- if (unit === 'count') {
- renderedValue = (
- <NumberContainer align={align}>
- {formatAbbreviatedNumber(typeof value === 'string' ? parseInt(value, 10) : value)}
- </NumberContainer>
- );
- }
- if (unit === CurrencyUnit.USD) {
- const numericValue = typeof value === 'string' ? parseFloat(value) : value;
- if (numericValue < 0.01) {
- renderedValue = (
- <NumberContainer align={align}>US {numericValue * 100}¢</NumberContainer>
- );
- } else if (numericValue >= 1) {
- renderedValue = (
- <NumberContainer align={align}>
- US ${formatAbbreviatedNumber(numericValue)}
- </NumberContainer>
- );
- }
- }
- if (unit === 'percentage') {
- renderedValue = (
- <NumberContainer align={align}>
- {formatPercentage(
- typeof value === 'string' ? parseFloat(value) : value,
- undefined,
- {minimumValue: MINIMUM_PERCENTAGE_VALUE}
- )}
- </NumberContainer>
- );
- }
- if (unit === 'percent_change') {
- renderedValue = (
- <NumberContainer align={align}>
- <PercentChange
- value={typeof value === 'string' ? parseFloat(value) : value}
- minimumValue={MINIMUM_PERCENTAGE_VALUE}
- />
- </NumberContainer>
- );
- }
- if (tooltip) {
- return (
- <NumberContainer align={align}>
- <Tooltip title={tooltip} isHoverable showUnderline>
- {renderedValue}
- </Tooltip>
- </NumberContainer>
- );
- }
- return <NumberContainer align={align}>{renderedValue}</NumberContainer>;
- }
- const MINIMUM_RATE_VALUE = 0.01;
- const MINIMUM_PERCENTAGE_VALUE = 0.0001; // 0.01%
- const NumberContainer = styled('div')<{align: 'left' | 'right'}>`
- text-align: ${p => p.align};
- font-variant-numeric: tabular-nums;
- `;
- const LoadingContainer = styled('div')<{align: 'left' | 'right'}>`
- display: flex;
- justify-content: ${p => (p.align === 'right' ? 'flex-end' : 'flex-start')};
- align-items: center;
- `;
- function isARateUnit(unit: string): unit is RateUnit {
- return (Object.values(RateUnit) as string[]).includes(unit);
- }
|