import styled from '@emotion/styled';
import {Tooltip} from 'sentry/components/tooltip';
import {IconFile} from 'sentry/icons';
import {t} from 'sentry/locale';
import {space} from 'sentry/styles/space';
import getDuration from 'sentry/utils/duration/getDuration';
import {useResourcesQuery} from 'sentry/views/insights/browser/common/queries/useResourcesQuery';
import type {WebVitals} from 'sentry/views/insights/browser/webVitals/types';
import {SpanMetricsField} from 'sentry/views/insights/types';
export function Recommendations({
transaction,
webVital,
}: {
transaction: string;
webVital: WebVitals;
}) {
switch (webVital) {
case 'lcp':
return null;
case 'cls':
return null;
case 'fcp':
return ;
case 'ttfb':
return null;
default:
return null;
}
}
function FcpRecommendations({transaction}: {transaction: string}) {
const query = `transaction:"${transaction}" resource.render_blocking_status:blocking`;
const {data, isLoading} = useResourcesQuery({
query,
sort: {field: `avg(${SpanMetricsField.SPAN_SELF_TIME})`, kind: 'desc'},
defaultResourceTypes: ['resource.script', 'resource.css', 'resource.img'],
limit: 7,
referrer: 'api.performance.browser.web-vitals.fcp-recommendations',
});
if (isLoading || !data || data.length < 1) {
return null;
}
return (
{t('Eliminate render blocking resources')}
{data.map(
({
'span.op': op,
'span.description': description,
'avg(span.self_time)': duration,
}) => {
return (
{description}
{getFormattedDuration(duration)}
);
}
)}
);
}
function RecommendationsHeader() {
return (
{t('Recommendations')}
);
}
function ResourceType({resourceType}: {resourceType: `resource.${string}`}) {
switch (resourceType) {
case 'resource.script':
return (
{t('js')}
{' \u2014 '}
);
case 'resource.css':
return (
{t('css')}
{' \u2014 '}
);
case 'resource.img':
return (
{t('img')}
{' \u2014 '}
);
default:
return null;
}
}
const getFormattedDuration = (value: number | null) => {
if (value === null) {
return null;
}
if (value < 1000) {
return getDuration(value / 1000, 0, true);
}
return getDuration(value / 1000, 2, true);
};
const StyledIconFile = styled(IconFile)`
margin-right: ${space(0.5)};
`;
const RecommendationSubHeader = styled('li')`
margin-bottom: ${space(1)};
`;
const RecommendationsHeaderContainer = styled('div')`
margin-bottom: ${space(1)};
font-size: ${p => p.theme.fontSizeExtraLarge};
`;
const ResourceList = styled('ul')`
padding-left: ${space(1)};
`;
const ResourceListItem = styled('li')`
margin-bottom: ${space(0.5)};
list-style: none;
white-space: nowrap;
`;
const ResourceDescription = styled('span')`
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
`;
const Flex = styled('span')`
display: flex;
justify-content: space-between;
gap: ${space(1)};
`;
const RecommendationsContainer = styled('div')`
margin-bottom: ${space(4)};
`;
const StyledTooltip = styled(Tooltip)`
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
`;