import {useMemo} from 'react'; import styled from '@emotion/styled'; import color from 'color'; import type {Location} from 'history'; import partition from 'lodash/partition'; import GuideAnchor from 'sentry/components/assistant/guideAnchor'; import {Button} from 'sentry/components/button'; import Collapsible from 'sentry/components/collapsible'; import GlobalSelectionLink from 'sentry/components/globalSelectionLink'; import Panel from 'sentry/components/panels/panel'; import PanelHeader from 'sentry/components/panels/panelHeader'; import TextOverflow from 'sentry/components/textOverflow'; import TimeSince from 'sentry/components/timeSince'; import {Tooltip} from 'sentry/components/tooltip'; import Version from 'sentry/components/version'; import {t, tct, tn} 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 {Release} from 'sentry/types/release'; import type {ReleasesDisplayOption} from '../releasesDisplayOptions'; import type {ReleasesRequestRenderProps} from '../releasesRequest'; import ReleaseCardCommits from './releaseCardCommits'; import ReleaseCardProjectRow from './releaseCardProjectRow'; import ReleaseCardStatsPeriod from './releaseCardStatsPeriod'; function getReleaseProjectId(release: Release, selection: PageFilters) { // if a release has only one project if (release.projects.length === 1) { return release.projects[0].id; } // if only one project is selected in global header and release has it (second condition will prevent false positives like -1) if ( selection.projects.length === 1 && release.projects.map(p => p.id).includes(selection.projects[0]) ) { return selection.projects[0]; } // project selector on release detail page will pick it up return undefined; } type Props = { activeDisplay: ReleasesDisplayOption; getHealthData: ReleasesRequestRenderProps['getHealthData']; isTopRelease: boolean; location: Location; organization: Organization; release: Release; reloading: boolean; selection: PageFilters; showHealthPlaceholders: boolean; showReleaseAdoptionStages: boolean; }; function ReleaseCard({ release, organization, activeDisplay, location, reloading, selection, showHealthPlaceholders, isTopRelease, getHealthData, showReleaseAdoptionStages, }: Props) { const { version, commitCount, lastDeploy, dateCreated, versionInfo, adoptionStages, projects, } = release; const [projectsToShow, projectsToHide] = useMemo(() => { // sort health rows inside release card alphabetically by project name, // show only the ones that are selected in global header return partition( projects.sort((a, b) => a.slug.localeCompare(b.slug)), p => // do not filter for My Projects & All Projects selection.projects.length > 0 && !selection.projects.includes(-1) ? selection.projects.includes(p.id) : true ); }, [projects, selection.projects]); const getHiddenProjectsTooltip = () => { const limitedProjects = projectsToHide.map(p => p.slug).slice(0, 5); const remainderLength = projectsToHide.length - limitedProjects.length; if (remainderLength) { limitedProjects.push(tn('and %s more', 'and %s more', remainderLength)); } return limitedProjects.join(', '); }; return ( {/* Header/info is the table sidecard */} 1} target="release_version" > {commitCount > 0 && ( )} {versionInfo?.package && ( {versionInfo.package} )} {lastDeploy?.dateFinished && ` \u007C ${lastDeploy.environment}`} {/* projects is the table */} {t('Project Name')} {showReleaseAdoptionStages && ( {t('Adoption Stage')} )} {t('Adoption')} {t('Crash Free Rate')} {t('Crashes')} {t('New Issues')} ( )} collapseButton={({onCollapse}) => ( )} > {projectsToShow.map((project, index) => { const key = `${project.slug}-${version}`; return ( ); })} {projectsToHide.length > 0 && ( {projectsToHide.length === 1 ? tct('[number:1] hidden project', {number: }) : tct('[number] hidden projects', { number: {projectsToHide.length}, })} )} ); } export const VersionWrapper = styled('div')` display: flex; align-items: center; `; const StyledVersion = styled(Version)` ${p => p.theme.overflowEllipsis}; `; const StyledPanel = styled(Panel)<{reloading: number}>` opacity: ${p => (p.reloading ? 0.5 : 1)}; pointer-events: ${p => (p.reloading ? 'none' : 'auto')}; @media (min-width: ${p => p.theme.breakpoints.medium}) { display: flex; } `; const ReleaseInfo = styled('div')` padding: ${space(1.5)} ${space(2)}; flex-shrink: 0; @media (min-width: ${p => p.theme.breakpoints.medium}) { border-right: 1px solid ${p => p.theme.border}; min-width: 260px; width: 22%; max-width: 300px; } `; export const ReleaseInfoSubheader = styled('div')` font-size: ${p => p.theme.fontSizeSmall}; color: ${p => p.theme.gray400}; `; export const PackageName = styled('div')` font-size: ${p => p.theme.fontSizeMedium}; color: ${p => p.theme.textColor}; display: flex; align-items: center; gap: ${space(0.5)}; `; const ReleaseProjects = styled('div')` border-top: 1px solid ${p => p.theme.border}; flex-grow: 1; display: grid; @media (min-width: ${p => p.theme.breakpoints.medium}) { border-top: none; } `; export const ReleaseInfoHeader = styled('div')` font-size: ${p => p.theme.fontSizeExtraLarge}; display: grid; grid-template-columns: minmax(0, 1fr) max-content; gap: ${space(2)}; align-items: center; `; const ReleaseProjectsHeader = styled(PanelHeader)` border-top-left-radius: 0; padding: ${space(1.5)} ${space(2)}; font-size: ${p => p.theme.fontSizeSmall}; `; const ProjectRows = styled('div')` position: relative; `; const ExpandButtonWrapper = styled('div')` position: absolute; width: 100%; bottom: 0; display: flex; align-items: center; justify-content: center; background-image: linear-gradient( 180deg, ${p => color(p.theme.background).alpha(0).string()} 0, ${p => p.theme.background} ); background-repeat: repeat-x; border-bottom: ${space(1)} solid ${p => p.theme.background}; border-top: ${space(1)} solid transparent; border-bottom-right-radius: ${p => p.theme.borderRadius}; @media (max-width: ${p => p.theme.breakpoints.medium}) { border-bottom-left-radius: ${p => p.theme.borderRadius}; } `; const CollapseButtonWrapper = styled('div')` display: flex; align-items: center; justify-content: center; height: 41px; `; export const ReleaseProjectsLayout = styled('div')<{ showReleaseAdoptionStages?: boolean; }>` display: grid; grid-template-columns: 1fr 1.4fr 0.6fr 0.7fr; grid-column-gap: ${space(1)}; align-items: center; width: 100%; @media (min-width: ${p => p.theme.breakpoints.small}) { grid-template-columns: 1fr 1fr 1fr 0.5fr 0.5fr 0.5fr; } @media (min-width: ${p => p.theme.breakpoints.medium}) { grid-template-columns: 1fr 1fr 1fr 0.5fr 0.5fr 0.5fr; } @media (min-width: ${p => p.theme.breakpoints.xlarge}) { ${p => { const adoptionStagesSize = p.showReleaseAdoptionStages ? '0.7fr' : ''; return `grid-template-columns: 1fr ${adoptionStagesSize} 1fr 1fr 0.7fr 0.7fr 0.5fr`; }} } `; export const ReleaseProjectColumn = styled('div')` ${p => p.theme.overflowEllipsis}; line-height: 20px; `; export const NewIssuesColumn = styled(ReleaseProjectColumn)` font-variant-numeric: tabular-nums; @media (min-width: ${p => p.theme.breakpoints.small}) { text-align: right; } `; export const AdoptionColumn = styled(ReleaseProjectColumn)` display: none; font-variant-numeric: tabular-nums; @media (min-width: ${p => p.theme.breakpoints.small}) { display: flex; /* Chart tooltips need overflow */ overflow: visible; } & > * { flex: 1; } `; export const AdoptionStageColumn = styled(ReleaseProjectColumn)` display: none; font-variant-numeric: tabular-nums; @media (min-width: ${p => p.theme.breakpoints.xlarge}) { display: flex; /* Need to show the edges of the tags */ overflow: visible; } `; export const CrashFreeRateColumn = styled(ReleaseProjectColumn)` font-variant-numeric: tabular-nums; @media (min-width: ${p => p.theme.breakpoints.small}) { text-align: center; } @media (min-width: ${p => p.theme.breakpoints.xlarge}) { text-align: right; } `; export const DisplaySmallCol = styled(ReleaseProjectColumn)` display: none; font-variant-numeric: tabular-nums; @media (min-width: ${p => p.theme.breakpoints.small}) { display: block; text-align: right; } `; const HiddenProjectsMessage = styled('div')` display: flex; align-items: center; font-size: ${p => p.theme.fontSizeSmall}; padding: 0 ${space(2)}; border-top: 1px solid ${p => p.theme.border}; overflow: hidden; height: 24px; line-height: 24px; color: ${p => p.theme.gray300}; background-color: ${p => p.theme.backgroundSecondary}; border-bottom-right-radius: ${p => p.theme.borderRadius}; @media (max-width: ${p => p.theme.breakpoints.medium}) { border-bottom-left-radius: ${p => p.theme.borderRadius}; } `; export default ReleaseCard;