123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210 |
- import {Fragment, useCallback, useRef, useState} from 'react';
- import styled from '@emotion/styled';
- import ErrorPanel from 'sentry/components/charts/errorPanel';
- import LoadingIndicator from 'sentry/components/loadingIndicator';
- import Placeholder from 'sentry/components/placeholder';
- import {IconWarning} from 'sentry/icons/iconWarning';
- import {space} from 'sentry/styles/space';
- import type {Organization} from 'sentry/types/organization';
- import {trackAnalytics} from 'sentry/utils/analytics';
- import getDynamicText from 'sentry/utils/getDynamicText';
- import {MEPDataProvider} from 'sentry/utils/performance/contexts/metricsEnhancedPerformanceDataContext';
- import useApi from 'sentry/utils/useApi';
- import getPerformanceWidgetContainer from 'sentry/views/performance/landing/widgets/components/performanceWidgetContainer';
- import type {
- GenericPerformanceWidgetProps,
- WidgetDataConstraint,
- WidgetDataProps,
- WidgetDataResult,
- WidgetPropUnion,
- } from '../types';
- import type {PerformanceWidgetSetting} from '../widgetDefinitions';
- import {DataStateSwitch} from './dataStateSwitch';
- import {QueryHandler} from './queryHandler';
- import {WidgetHeader} from './widgetHeader';
- // Generic performance widget for type T, where T defines all the data contained in the widget.
- export function GenericPerformanceWidget<T extends WidgetDataConstraint>(
- props: WidgetPropUnion<T>
- ) {
- // Use object keyed to chart setting so switching between charts of a similar type doesn't retain data with query components still having inflight requests.
- const [allWidgetData, setWidgetData] = useState<{[chartSetting: string]: T}>({});
- const widgetData = allWidgetData[props.chartSetting] ?? {};
- const widgetDataRef = useRef(widgetData);
- const setWidgetDataForKey = useCallback(
- (dataKey: string, result?: WidgetDataResult) => {
- const _widgetData = widgetDataRef.current;
- const newWidgetData = {..._widgetData, [dataKey]: result};
- widgetDataRef.current = newWidgetData;
- setWidgetData({[props.chartSetting]: newWidgetData});
- },
- // eslint-disable-next-line react-hooks/exhaustive-deps
- [allWidgetData, setWidgetData]
- );
- const removeWidgetDataForKey = useCallback(
- (dataKey: string) => {
- const _widgetData = widgetDataRef.current;
- const newWidgetData = {..._widgetData};
- delete newWidgetData[dataKey];
- widgetDataRef.current = newWidgetData;
- setWidgetData({[props.chartSetting]: newWidgetData});
- },
- // eslint-disable-next-line react-hooks/exhaustive-deps
- [allWidgetData, setWidgetData]
- );
- const widgetProps = {widgetData, setWidgetDataForKey, removeWidgetDataForKey};
- const queries = Object.entries(props.Queries).map(([key, definition]) => ({
- ...definition,
- queryKey: key,
- }));
- const api = useApi();
- const totalHeight = props.Visualizations.reduce((acc, curr) => acc + curr.height, 0);
- return (
- <Fragment>
- <MEPDataProvider chartSetting={props.chartSetting}>
- <QueryHandler
- eventView={props.eventView}
- widgetData={widgetData}
- setWidgetDataForKey={setWidgetDataForKey}
- removeWidgetDataForKey={removeWidgetDataForKey}
- queryProps={props}
- queries={queries}
- api={api}
- />
- <DataDisplay<T> {...props} {...widgetProps} totalHeight={totalHeight} />
- </MEPDataProvider>
- </Fragment>
- );
- }
- function trackDataComponentClicks(
- chartSetting: PerformanceWidgetSetting,
- organization: Organization
- ) {
- trackAnalytics('performance_views.landingv3.widget.interaction', {
- organization,
- widget_type: chartSetting,
- });
- }
- export function DataDisplay<T extends WidgetDataConstraint>(
- props: GenericPerformanceWidgetProps<T> & WidgetDataProps<T> & {totalHeight: number}
- ) {
- const {Visualizations, chartHeight, totalHeight, containerType, EmptyComponent} = props;
- const Container = getPerformanceWidgetContainer({
- containerType,
- });
- const numberKeys = Object.keys(props.Queries).length;
- const missingDataKeys = Object.values(props.widgetData).length !== numberKeys;
- const hasData =
- !missingDataKeys && Object.values(props.widgetData).every(d => !d || d.hasData);
- const isLoading = Object.values(props.widgetData).some(d => !d || d.isLoading);
- const isErrored =
- !missingDataKeys && Object.values(props.widgetData).some(d => d?.isErrored);
- return (
- <Container data-test-id="performance-widget-container">
- <ContentContainer>
- <WidgetHeader<T> {...props} />
- </ContentContainer>
- <DataStateSwitch
- isLoading={isLoading}
- isErrored={isErrored}
- hasData={hasData}
- errorComponent={<DefaultErrorComponent height={totalHeight} />}
- dataComponents={Visualizations.map((Visualization, index) => (
- <ContentBodyContainer
- key={index}
- noPadding={Visualization.noPadding}
- bottomPadding={Visualization.bottomPadding}
- data-test-id="widget-state-has-data"
- onClick={() =>
- trackDataComponentClicks(props.chartSetting, props.organization)
- }
- >
- {getDynamicText({
- value: (
- <Visualization.component
- grid={defaultGrid}
- queryFields={Visualization.fields}
- widgetData={props.widgetData}
- height={chartHeight}
- />
- ),
- fixed: <Placeholder height={`${chartHeight}px`} />,
- })}
- </ContentBodyContainer>
- ))}
- loadingComponent={
- <LoadingWrapper height={totalHeight}>
- <StyledLoadingIndicator size={40} />
- </LoadingWrapper>
- }
- emptyComponent={
- EmptyComponent ? (
- <EmptyComponent />
- ) : (
- <PerformanceWidgetPlaceholder height={`${totalHeight}px`} />
- )
- }
- />
- </Container>
- );
- }
- function DefaultErrorComponent(props: {height: number}) {
- return (
- <ErrorPanel data-test-id="widget-state-is-errored" height={`${props.height}px`}>
- <IconWarning color="gray300" size="lg" />
- </ErrorPanel>
- );
- }
- const defaultGrid = {
- left: 0,
- right: 0,
- top: space(2),
- bottom: space(1),
- };
- const ContentContainer = styled('div')<{bottomPadding?: boolean; noPadding?: boolean}>`
- padding-left: ${p => (p.noPadding ? 0 : space(2))};
- padding-right: ${p => (p.noPadding ? 0 : space(2))};
- padding-bottom: ${p => (p.bottomPadding ? space(1) : 0)};
- `;
- const ContentBodyContainer = styled(ContentContainer)`
- height: 100%;
- `;
- const PerformanceWidgetPlaceholder = styled(Placeholder)`
- border-color: transparent;
- border-bottom-right-radius: inherit;
- border-bottom-left-radius: inherit;
- `;
- const LoadingWrapper = styled('div')<{height?: number}>`
- height: ${p => p.height}px;
- display: flex;
- align-items: center;
- justify-content: center;
- `;
- const StyledLoadingIndicator = styled(LoadingIndicator)`
- margin: 0;
- `;
- GenericPerformanceWidget.defaultProps = {
- containerType: 'panel',
- chartHeight: 200,
- };
|