123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320 |
- import type {FC} from 'react';
- import {Fragment, useEffect, useRef} from 'react';
- import type {InjectedRouter} from 'react-router';
- import styled from '@emotion/styled';
- import type {Location} from 'history';
- import {Button} from 'sentry/components/button';
- import ButtonBar from 'sentry/components/buttonBar';
- import FeedbackWidgetButton from 'sentry/components/feedback/widget/feedbackWidgetButton';
- import * as Layout from 'sentry/components/layouts/thirds';
- import LoadingIndicator from 'sentry/components/loadingIndicator';
- import {DatePageFilter} from 'sentry/components/organizations/datePageFilter';
- import {EnvironmentPageFilter} from 'sentry/components/organizations/environmentPageFilter';
- import PageFilterBar from 'sentry/components/organizations/pageFilterBar';
- import {ProjectPageFilter} from 'sentry/components/organizations/projectPageFilter';
- import {PageHeadingQuestionTooltip} from 'sentry/components/pageHeadingQuestionTooltip';
- import TransactionNameSearchBar from 'sentry/components/performance/searchBar';
- import * as TeamKeyTransactionManager from 'sentry/components/performance/teamKeyTransactionsManager';
- import {TabList, TabPanels, Tabs} from 'sentry/components/tabs';
- import {t} from 'sentry/locale';
- import {space} from 'sentry/styles/space';
- import type {PageFilters} from 'sentry/types/core';
- import type {Organization} from 'sentry/types/organization';
- import type {Project} from 'sentry/types/project';
- import {trackAnalytics} from 'sentry/utils/analytics';
- import {browserHistory} from 'sentry/utils/browserHistory';
- import type EventView from 'sentry/utils/discover/eventView';
- import {GenericQueryBatcher} from 'sentry/utils/performance/contexts/genericQueryBatcher';
- import {MetricsCardinalityProvider} from 'sentry/utils/performance/contexts/metricsCardinality';
- import type {MEPState} from 'sentry/utils/performance/contexts/metricsEnhancedSetting';
- import {
- MEPConsumer,
- MEPSettingProvider,
- } from 'sentry/utils/performance/contexts/metricsEnhancedSetting';
- import {PageAlert, PageAlertProvider} from 'sentry/utils/performance/contexts/pageAlert';
- import {MutableSearch} from 'sentry/utils/tokenizeSearch';
- import {useTeams} from 'sentry/utils/useTeams';
- import Onboarding from '../onboarding';
- import {MetricsEventsDropdown} from '../transactionSummary/transactionOverview/metricEvents/metricsEventsDropdown';
- import {getTransactionSearchQuery} from '../utils';
- import {AllTransactionsView} from './views/allTransactionsView';
- import {BackendView} from './views/backendView';
- import {FrontendOtherView} from './views/frontendOtherView';
- import {FrontendPageloadView} from './views/frontendPageloadView';
- import {MobileView} from './views/mobileView';
- import {MetricsDataSwitcher} from './metricsDataSwitcher';
- import {MetricsDataSwitcherAlert} from './metricsDataSwitcherAlert';
- import {
- getDefaultDisplayForPlatform,
- getLandingDisplayFromParam,
- handleLandingDisplayChange,
- LANDING_DISPLAYS,
- LandingDisplayField,
- } from './utils';
- type Props = {
- eventView: EventView;
- handleSearch: (searchQuery: string, currentMEPState?: MEPState) => void;
- handleTrendsClick: () => void;
- location: Location;
- onboardingProject: Project | undefined;
- organization: Organization;
- projects: Project[];
- router: InjectedRouter;
- selection: PageFilters;
- setError: (msg: string | undefined) => void;
- withStaticFilters: boolean;
- };
- const fieldToViewMap: Record<LandingDisplayField, FC<Props>> = {
- [LandingDisplayField.ALL]: AllTransactionsView,
- [LandingDisplayField.BACKEND]: BackendView,
- [LandingDisplayField.FRONTEND_OTHER]: FrontendOtherView,
- [LandingDisplayField.FRONTEND_PAGELOAD]: FrontendPageloadView,
- [LandingDisplayField.MOBILE]: MobileView,
- };
- export function PerformanceLanding(props: Props) {
- const {
- organization,
- location,
- eventView,
- projects,
- handleSearch,
- handleTrendsClick,
- onboardingProject,
- } = props;
- const {teams, initiallyLoaded} = useTeams({provideUserTeams: true});
- const hasMounted = useRef(false);
- const paramLandingDisplay = getLandingDisplayFromParam(location);
- const defaultLandingDisplayForProjects = getDefaultDisplayForPlatform(
- projects,
- eventView
- );
- const landingDisplay = paramLandingDisplay ?? defaultLandingDisplayForProjects;
- const showOnboarding = onboardingProject !== undefined;
- useEffect(() => {
- if (hasMounted.current) {
- browserHistory.replace({
- pathname: location.pathname,
- query: {
- ...location.query,
- landingDisplay: undefined,
- },
- });
- }
- // eslint-disable-next-line react-hooks/exhaustive-deps
- }, [eventView.project.join('.')]);
- useEffect(() => {
- hasMounted.current = true;
- }, []);
- useEffect(() => {
- if (showOnboarding) {
- trackAnalytics('performance_views.overview.has_data', {
- table_data_state: 'onboarding',
- tab: paramLandingDisplay?.field,
- organization,
- });
- }
- }, [showOnboarding, paramLandingDisplay, organization]);
- const getFreeTextFromQuery = (query: string) => {
- const conditions = new MutableSearch(query);
- const transactionValues = conditions.getFilterValues('transaction');
- if (transactionValues.length) {
- return transactionValues[0];
- }
- if (conditions.freeText.length > 0) {
- // raw text query will be wrapped in wildcards in generatePerformanceEventView
- // so no need to wrap it here
- return conditions.freeText.join(' ');
- }
- return '';
- };
- const derivedQuery = getTransactionSearchQuery(location, eventView.query);
- const ViewComponent = fieldToViewMap[landingDisplay.field];
- let pageFilters: React.ReactNode = (
- <PageFilterBar condensed>
- <ProjectPageFilter />
- <EnvironmentPageFilter />
- <DatePageFilter />
- </PageFilterBar>
- );
- if (showOnboarding) {
- pageFilters = <SearchContainerWithFilter>{pageFilters}</SearchContainerWithFilter>;
- }
- const SearchFilterContainer = organization.features.includes('performance-use-metrics')
- ? SearchContainerWithFilterAndMetrics
- : SearchContainerWithFilter;
- return (
- <Layout.Page data-test-id="performance-landing-v3">
- <PageAlertProvider>
- <Tabs
- value={landingDisplay.field}
- onChange={field =>
- handleLandingDisplayChange(field, location, projects, organization, eventView)
- }
- >
- <Layout.Header>
- <Layout.HeaderContent>
- <Layout.Title>
- {t('Performance')}
- <PageHeadingQuestionTooltip
- docsUrl="https://docs.sentry.io/product/performance/"
- title={t(
- 'Your main view for transaction data with graphs that visualize transactions or trends, as well as a table where you can drill down on individual transactions.'
- )}
- />
- </Layout.Title>
- </Layout.HeaderContent>
- <Layout.HeaderActions>
- {!showOnboarding && (
- <ButtonBar gap={1}>
- <Button
- size="sm"
- priority="primary"
- data-test-id="landing-header-trends"
- onClick={() => handleTrendsClick()}
- >
- {t('View Trends')}
- </Button>
- <FeedbackWidgetButton />
- </ButtonBar>
- )}
- </Layout.HeaderActions>
- <TabList hideBorder>
- {LANDING_DISPLAYS.map(({label, field}) => (
- <TabList.Item key={field}>{label}</TabList.Item>
- ))}
- </TabList>
- </Layout.Header>
- <Layout.Body data-test-id="performance-landing-body">
- <Layout.Main fullWidth>
- <TabPanels>
- <TabPanels.Item key={landingDisplay.field}>
- <MetricsCardinalityProvider
- sendOutcomeAnalytics
- organization={organization}
- location={location}
- >
- <MetricsDataSwitcher
- organization={organization}
- eventView={eventView}
- location={location}
- >
- {metricsDataSide => {
- return (
- <MEPSettingProvider
- location={location}
- forceTransactions={metricsDataSide.forceTransactionsOnly}
- >
- <MetricsDataSwitcherAlert
- organization={organization}
- eventView={eventView}
- projects={projects}
- location={location}
- router={props.router}
- {...metricsDataSide}
- />
- <PageAlert />
- {showOnboarding ? (
- <Fragment>
- {pageFilters}
- <Onboarding
- organization={organization}
- project={onboardingProject}
- />
- </Fragment>
- ) : (
- <Fragment>
- <SearchFilterContainer>
- {pageFilters}
- <MEPConsumer>
- {({metricSettingState}) => (
- // TODO replace `handleSearch prop` with transaction name search once
- // transaction name search becomes the default search bar
- <TransactionNameSearchBar
- organization={organization}
- eventView={eventView}
- onSearch={(query: string) => {
- handleSearch(
- query,
- metricSettingState ?? undefined
- );
- }}
- query={getFreeTextFromQuery(derivedQuery)}
- />
- )}
- </MEPConsumer>
- <MetricsEventsDropdown />
- </SearchFilterContainer>
- {initiallyLoaded ? (
- <TeamKeyTransactionManager.Provider
- organization={organization}
- teams={teams}
- selectedTeams={['myteams']}
- selectedProjects={eventView.project.map(String)}
- >
- <GenericQueryBatcher>
- <ViewComponent {...props} />
- </GenericQueryBatcher>
- </TeamKeyTransactionManager.Provider>
- ) : (
- <LoadingIndicator />
- )}
- </Fragment>
- )}
- </MEPSettingProvider>
- );
- }}
- </MetricsDataSwitcher>
- </MetricsCardinalityProvider>
- </TabPanels.Item>
- </TabPanels>
- </Layout.Main>
- </Layout.Body>
- </Tabs>
- </PageAlertProvider>
- </Layout.Page>
- );
- }
- const SearchContainerWithFilter = styled('div')`
- display: grid;
- grid-template-rows: auto auto;
- gap: ${space(2)};
- margin-bottom: ${space(2)};
- @media (min-width: ${p => p.theme.breakpoints.small}) {
- grid-template-rows: auto;
- grid-template-columns: auto 1fr;
- }
- `;
- const SearchContainerWithFilterAndMetrics = styled('div')`
- display: grid;
- grid-template-rows: auto auto auto;
- gap: ${space(2)};
- margin-bottom: ${space(2)};
- @media (min-width: ${p => p.theme.breakpoints.small}) {
- grid-template-rows: auto;
- grid-template-columns: auto 1fr auto;
- }
- `;
|