import {Fragment} from 'react'; import styled from '@emotion/styled'; import {motion, Variants} from 'framer-motion'; import {PlatformIcon} from 'platformicons'; import {PlatformKey} from 'sentry/data/platformCategories'; import platforms from 'sentry/data/platforms'; import {IconCheckmark} from 'sentry/icons'; import {t} from 'sentry/locale'; import pulsingIndicatorStyles from 'sentry/styles/pulsingIndicator'; import {space} from 'sentry/styles/space'; import {Project} from 'sentry/types'; import testableTransition from 'sentry/utils/testableTransition'; type Props = { activeProject: Project | null; checkProjectHasFirstEvent: (project: Project) => boolean; projects: Project[]; selectProject: (newProjectId: string) => void; // A map from selected platform keys to the projects created by onboarding. selectedPlatformToProjectIdMap: {[key in PlatformKey]?: string}; }; function ProjectSidebarSection({ projects, activeProject, selectProject, checkProjectHasFirstEvent, selectedPlatformToProjectIdMap, }: Props) { const oneProject = (platformOnCreate: string, projectSlug: string) => { const project = projects.find(p => p.slug === projectSlug); const platform = project ? project.platform || 'other' : platformOnCreate; const platformName = platforms.find(p => p.id === platform)?.name ?? ''; const isActive = !!project && activeProject?.id === project.id; const errorReceived = !!project && checkProjectHasFirstEvent(project); return ( project && selectProject(project.id)} disabled={!project} > {platformName} {!project ? t('Project Deleted') : errorReceived ? t('Error Received') : t('Waiting for error')} {errorReceived ? ( ) : ( isActive && )} ); }; return ( {t('Projects to Setup')} {Object.entries(selectedPlatformToProjectIdMap).map( ([platformOnCreate, projectSlug]) => { return oneProject(platformOnCreate, projectSlug); } )} ); } export default ProjectSidebarSection; const Title = styled('span')` font-size: 12px; font-weight: 600; text-transform: uppercase; margin-left: ${space(2)}; `; const SubHeader = styled('div')<{errorReceived: boolean}>` color: ${p => (p.errorReceived ? p.theme.successText : p.theme.pink400)}; `; const StyledPlatformIcon = styled(PlatformIcon)``; const ProjectWrapper = styled('div')<{disabled: boolean; isActive: boolean}>` display: flex; flex-direction: row; align-items: center; background-color: ${p => p.isActive && p.theme.gray100}; padding: ${space(2)}; cursor: pointer; border-radius: 4px; user-select: none; ${p => p.disabled && ` cursor: not-allowed; ${StyledPlatformIcon} { filter: grayscale(1); } ${SubHeader} { color: ${p.theme.gray400}; } ${Beat} { color: ${p.theme.gray400}; } ${NameWrapper} { text-decoration-line: line-through; } `} `; const indicatorAnimation: Variants = { initial: {opacity: 0, y: -10}, animate: {opacity: 1, y: 0}, exit: {opacity: 0, y: 10}, }; const WaitingIndicator = styled(motion.div)` margin: 0 6px; flex-shrink: 0; ${pulsingIndicatorStyles}; background-color: ${p => p.theme.pink300}; `; const StyledIconCheckmark = styled(IconCheckmark)` flex-shrink: 0; `; WaitingIndicator.defaultProps = { variants: indicatorAnimation, transition: testableTransition(), }; const MiddleWrapper = styled('div')` margin: 0 ${space(1)}; flex-grow: 1; overflow: hidden; `; const NameWrapper = styled('div')` overflow: hidden; white-space: nowrap; text-overflow: ellipsis; `; const Beat = styled('div')<{color: string}>` color: ${p => p.color}; `;