metricsRibbon.tsx 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147
  1. import type {ComponentProps} from 'react';
  2. import {useMemo} from 'react';
  3. import styled from '@emotion/styled';
  4. import {space} from 'sentry/styles/space';
  5. import type {NewQuery} from 'sentry/types/organization';
  6. import type {Project} from 'sentry/types/project';
  7. import type {TableData, TableDataRow} from 'sentry/utils/discover/discoverQuery';
  8. import EventView from 'sentry/utils/discover/eventView';
  9. import type {DiscoverDatasets} from 'sentry/utils/discover/types';
  10. import {decodeScalar} from 'sentry/utils/queryString';
  11. import {MutableSearch} from 'sentry/utils/tokenizeSearch';
  12. import {useLocation} from 'sentry/utils/useLocation';
  13. import useOrganization from 'sentry/utils/useOrganization';
  14. import usePageFilters from 'sentry/utils/usePageFilters';
  15. import {MetricReadout} from 'sentry/views/performance/metricReadout';
  16. import {
  17. DEFAULT_PLATFORM,
  18. PLATFORM_LOCAL_STORAGE_KEY,
  19. PLATFORM_QUERY_PARAM,
  20. } from 'sentry/views/performance/mobile/screenload/screens/platformSelector';
  21. import {useTableQuery} from 'sentry/views/performance/mobile/screenload/screens/screensTable';
  22. import {isCrossPlatform} from 'sentry/views/performance/mobile/screenload/screens/utils';
  23. import {useReleaseSelection} from 'sentry/views/starfish/queries/useReleases';
  24. import {appendReleaseFilters} from 'sentry/views/starfish/utils/releaseComparison';
  25. interface BlockProps {
  26. dataKey: string | ((data?: TableDataRow[]) => number | undefined);
  27. title: string;
  28. unit: ComponentProps<typeof MetricReadout>['unit'];
  29. allowZero?: boolean;
  30. }
  31. export function MetricsRibbon({
  32. filters,
  33. project,
  34. blocks,
  35. fields,
  36. referrer,
  37. dataset,
  38. }: {
  39. blocks: BlockProps[];
  40. dataset: DiscoverDatasets;
  41. fields: string[];
  42. referrer: string;
  43. filters?: string[];
  44. project?: Project | null;
  45. }) {
  46. const {selection} = usePageFilters();
  47. const organization = useOrganization();
  48. const location = useLocation();
  49. const {
  50. primaryRelease,
  51. secondaryRelease,
  52. isLoading: isReleasesLoading,
  53. } = useReleaseSelection();
  54. const hasPlatformSelectFeature = organization.features.includes('spans-first-ui');
  55. const platform =
  56. decodeScalar(location.query[PLATFORM_QUERY_PARAM]) ??
  57. localStorage.getItem(PLATFORM_LOCAL_STORAGE_KEY) ??
  58. DEFAULT_PLATFORM;
  59. const queryString = useMemo(() => {
  60. const searchQuery = new MutableSearch([...(filters ?? [])]);
  61. if (project && isCrossPlatform(project) && hasPlatformSelectFeature) {
  62. searchQuery.addFilterValue('os.name', platform);
  63. }
  64. return appendReleaseFilters(searchQuery, primaryRelease, secondaryRelease);
  65. }, [
  66. filters,
  67. hasPlatformSelectFeature,
  68. platform,
  69. primaryRelease,
  70. project,
  71. secondaryRelease,
  72. ]);
  73. const newQuery: NewQuery = {
  74. name: 'ScreenMetricsRibbon',
  75. fields,
  76. query: queryString,
  77. dataset,
  78. version: 2,
  79. projects: selection.projects,
  80. };
  81. const eventView = EventView.fromNewQueryWithLocation(newQuery, location);
  82. const {data, isLoading} = useTableQuery({
  83. eventView,
  84. enabled: !isReleasesLoading,
  85. referrer,
  86. });
  87. return (
  88. <BlockContainer>
  89. {blocks.map(({title, dataKey, unit}) => (
  90. <MetricsBlock
  91. key={title}
  92. title={title}
  93. unit={unit}
  94. dataKey={dataKey}
  95. data={data}
  96. isLoading={isLoading}
  97. />
  98. ))}
  99. </BlockContainer>
  100. );
  101. }
  102. function MetricsBlock({
  103. title,
  104. unit,
  105. data,
  106. dataKey,
  107. isLoading,
  108. allowZero,
  109. }: {
  110. isLoading: boolean;
  111. title: string;
  112. data?: TableData;
  113. release?: string;
  114. } & BlockProps) {
  115. const value =
  116. typeof dataKey === 'function'
  117. ? dataKey(data?.data)
  118. : (data?.data?.[0]?.[dataKey] as number);
  119. const hasData = (value && value !== 0) || (value === 0 && allowZero);
  120. return (
  121. <MetricReadout
  122. title={title}
  123. align="left"
  124. value={hasData ? value : undefined}
  125. isLoading={isLoading}
  126. unit={unit}
  127. />
  128. );
  129. }
  130. const BlockContainer = styled('div')`
  131. display: flex;
  132. gap: ${space(2)};
  133. `;