import {Fragment} from 'react';
import {useTheme} from '@emotion/react';
import styled from '@emotion/styled';
import InteractionStateLayer from 'sentry/components/interactionStateLayer';
import ExternalLink from 'sentry/components/links/externalLink';
import QuestionTooltip from 'sentry/components/questionTooltip';
import {Tooltip} from 'sentry/components/tooltip';
import {t, tct} from 'sentry/locale';
import {space} from 'sentry/styles/space';
import type {TableData} from 'sentry/utils/discover/discoverQuery';
import getDuration from 'sentry/utils/duration/getDuration';
import {MODULE_DOC_LINK} from 'sentry/views/insights/browser/webVitals/settings';
import type {
ProjectScore,
WebVitals,
} from 'sentry/views/insights/browser/webVitals/types';
import {PERFORMANCE_SCORE_COLORS} from 'sentry/views/insights/browser/webVitals/utils/performanceScoreColors';
import {
scoreToStatus,
STATUS_TEXT,
} from 'sentry/views/insights/browser/webVitals/utils/scoreToStatus';
type Props = {
onClick?: (webVital: WebVitals) => void;
projectData?: TableData;
projectScore?: ProjectScore;
showTooltip?: boolean;
transaction?: string;
};
const WEB_VITALS_METERS_CONFIG = {
lcp: {
name: t('Largest Contentful Paint'),
formatter: (value: number) => getFormattedDuration(value / 1000),
},
fcp: {
name: t('First Contentful Paint'),
formatter: (value: number) => getFormattedDuration(value / 1000),
},
inp: {
name: t('Interaction to Next Paint'),
formatter: (value: number) => getFormattedDuration(value / 1000),
},
cls: {
name: t('Cumulative Layout Shift'),
formatter: (value: number) => Math.round(value * 100) / 100,
},
ttfb: {
name: t('Time To First Byte'),
formatter: (value: number) => getFormattedDuration(value / 1000),
},
};
export default function WebVitalMeters({
onClick,
projectData,
projectScore,
showTooltip = true,
}: Props) {
const theme = useTheme();
if (!projectScore) {
return null;
}
const webVitalsConfig = WEB_VITALS_METERS_CONFIG;
const webVitals = Object.keys(webVitalsConfig) as WebVitals[];
const colors = theme.charts.getColorPalette(3);
return (
{webVitals.map((webVital, index) => {
const webVitalExists = projectScore[`${webVital}Score`] !== undefined;
const formattedMeterValueText = webVitalExists ? (
webVitalsConfig[webVital].formatter(
projectData?.data?.[0]?.[`p75(measurements.${webVital})`] as number
)
) : (
);
const headerText = webVitalsConfig[webVital].name;
const meterBody = (
{showTooltip && (
{tct(
`The p75 [webVital] value and aggregate [webVital] score of your selected project(s).
Scores and values may share some (but not perfect) correlation.`,
{
webVital: webVital.toUpperCase(),
}
)}
{t('Find out how performance scores are calculated here.')}
}
/>
)}
{headerText}
{formattedMeterValueText}
);
return (
);
})}
);
}
type VitalContainerProps = {
meterBody: React.ReactNode;
webVital: WebVitals;
webVitalExists: boolean;
onClick?: (webVital: WebVitals) => void;
};
function VitalContainer({
webVital,
webVitalExists,
meterBody,
onClick,
}: VitalContainerProps) {
return (
webVitalExists && onClick?.(webVital)}
clickable={webVitalExists}
>
{webVitalExists && }
{webVitalExists && meterBody}
{!webVitalExists && (
{meterBody}
)}
);
}
export const getFormattedDuration = (value: number) => {
return getDuration(value, value < 1 ? 0 : 2, true);
};
const Container = styled('div')`
margin-bottom: ${space(1)};
`;
const Flex = styled('div')<{gap?: number}>`
display: flex;
flex-direction: row;
justify-content: center;
width: 100%;
gap: ${p => (p.gap ? `${p.gap}px` : space(1))};
align-items: center;
flex-wrap: wrap;
`;
const MeterBarContainer = styled('div')<{clickable?: boolean}>`
flex: 1;
position: relative;
padding: 0;
cursor: ${p => (p.clickable ? 'pointer' : 'default')};
min-width: 140px;
`;
const MeterBarBody = styled('div')`
border: 1px solid ${p => p.theme.gray200};
border-radius: ${p => p.theme.borderRadius} ${p => p.theme.borderRadius} 0 0;
border-bottom: none;
padding: ${space(1)} 0 ${space(0.5)} 0;
`;
const MeterHeader = styled('div')`
font-size: ${p => p.theme.fontSizeSmall};
color: ${p => p.theme.textColor};
display: inline-block;
text-align: center;
width: 100%;
`;
const MeterValueText = styled('div')`
display: flex;
justify-content: center;
align-items: center;
font-size: ${p => p.theme.headerFontSize};
color: ${p => p.theme.textColor};
flex: 1;
text-align: center;
`;
function MeterBarFooter({score}: {score: number | undefined}) {
if (score === undefined) {
return (
{t('No Data')}
);
}
const status = scoreToStatus(score);
return (
{STATUS_TEXT[status]} {score}
);
}
const MeterBarFooterContainer = styled('div')<{status: string}>`
color: ${p => p.theme[PERFORMANCE_SCORE_COLORS[p.status].normal]};
border-radius: 0 0 ${p => p.theme.borderRadius} ${p => p.theme.borderRadius};
background-color: ${p => p.theme[PERFORMANCE_SCORE_COLORS[p.status].light]};
border: solid 1px ${p => p.theme[PERFORMANCE_SCORE_COLORS[p.status].light]};
font-size: ${p => p.theme.fontSizeExtraSmall};
padding: ${space(0.5)};
text-align: center;
`;
const NoValueContainer = styled('span')`
color: ${p => p.theme.gray300};
font-size: ${p => p.theme.headerFontSize};
`;
function NoValue() {
return {' \u2014 '};
}
const StyledTooltip = styled(Tooltip)`
display: block;
width: 100%;
`;
const StyledQuestionTooltip = styled(QuestionTooltip)`
position: absolute;
right: ${space(1)};
`;
export const Dot = styled('span')<{color: string}>`
display: inline-block;
margin-right: ${space(1)};
border-radius: ${p => p.theme.borderRadius};
width: ${space(1)};
height: ${space(1)};
background-color: ${p => p.color};
`;