123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140 |
- import {Fragment} from 'react';
- import styled from '@emotion/styled';
- import {Location} from 'history';
- import GuideAnchor from 'sentry/components/assistant/guideAnchor';
- import {SectionHeading} from 'sentry/components/charts/styles';
- import Link from 'sentry/components/links/link';
- import Placeholder from 'sentry/components/placeholder';
- import QuestionTooltip from 'sentry/components/questionTooltip';
- import UserMisery from 'sentry/components/userMisery';
- import {IconOpen} from 'sentry/icons';
- import {t} from 'sentry/locale';
- import {Organization} from 'sentry/types';
- import EventView from 'sentry/utils/discover/eventView';
- import {QueryError} from 'sentry/utils/discover/genericDiscoverQuery';
- import {WebVital} from 'sentry/utils/fields';
- import {useMEPSettingContext} from 'sentry/utils/performance/contexts/metricsEnhancedSetting';
- import {decodeScalar} from 'sentry/utils/queryString';
- import {getTermHelp, PERFORMANCE_TERM} from 'sentry/views/performance/data';
- import {getTransactionMEPParamsIfApplicable} from 'sentry/views/performance/transactionSummary/transactionOverview/utils';
- import {vitalsRouteWithQuery} from 'sentry/views/performance/transactionSummary/transactionVitals/utils';
- import {SidebarSpacer} from 'sentry/views/performance/transactionSummary/utils';
- import VitalInfo from 'sentry/views/performance/vitalDetail/vitalInfo';
- type Props = {
- error: QueryError | null;
- eventView: EventView;
- hasWebVitals: boolean;
- isLoading: boolean;
- location: Location;
- organization: Organization;
- totals: Record<string, number> | null;
- transactionName: string;
- };
- function UserStats({
- isLoading,
- hasWebVitals,
- error,
- totals,
- location,
- organization,
- transactionName,
- eventView,
- }: Props) {
- let userMisery = error !== null ? <div>{'\u2014'}</div> : <Placeholder height="34px" />;
- if (!isLoading && error === null && totals) {
- const threshold: number | undefined = totals.project_threshold_config
- ? totals.project_threshold_config[1]
- : undefined;
- const miserableUsers: number | undefined = totals['count_miserable_user()'];
- const userMiseryScore: number = totals['user_misery()'] || 0;
- const totalUsers = totals['count_unique_user()'];
- userMisery = (
- <UserMisery
- bars={40}
- barHeight={30}
- userMisery={userMiseryScore}
- miseryLimit={threshold}
- totalUsers={totalUsers}
- miserableUsers={miserableUsers}
- />
- );
- }
- const orgSlug = organization.slug;
- const webVitalsTarget = vitalsRouteWithQuery({
- orgSlug,
- transaction: transactionName,
- projectID: decodeScalar(location.query.project),
- query: location.query,
- });
- const mepSetting = useMEPSettingContext();
- const queryExtras = getTransactionMEPParamsIfApplicable(
- mepSetting,
- organization,
- location
- );
- return (
- <Fragment>
- {hasWebVitals && (
- <Fragment>
- <VitalsHeading>
- <SectionHeading>
- {t('Web Vitals')}
- <QuestionTooltip
- position="top"
- title={t(
- 'Web Vitals with p75 better than the "poor" threshold, as defined by Google Web Vitals.'
- )}
- size="sm"
- />
- </SectionHeading>
- <Link to={webVitalsTarget}>
- <IconOpen />
- </Link>
- </VitalsHeading>
- <VitalInfo
- location={location}
- vital={[WebVital.FCP, WebVital.LCP, WebVital.FID, WebVital.CLS]}
- orgSlug={orgSlug}
- environment={eventView.environment}
- start={eventView.start}
- end={eventView.end}
- statsPeriod={eventView.statsPeriod}
- project={eventView.project}
- hideVitalThresholds
- hideDurationDetail
- queryExtras={queryExtras}
- />
- <SidebarSpacer />
- </Fragment>
- )}
- <GuideAnchor target="user_misery" position="left">
- <SectionHeading>
- {t('User Misery')}
- <QuestionTooltip
- position="top"
- title={getTermHelp(organization, PERFORMANCE_TERM.USER_MISERY)}
- size="sm"
- />
- </SectionHeading>
- </GuideAnchor>
- {userMisery}
- <SidebarSpacer />
- </Fragment>
- );
- }
- const VitalsHeading = styled('div')`
- display: flex;
- justify-content: space-between;
- align-items: center;
- `;
- export default UserStats;
|