import {useCallback} from 'react'; import styled from '@emotion/styled'; import type {Variants} from 'framer-motion'; import {motion} from 'framer-motion'; import {Button, LinkButton} from 'sentry/components/button'; import ButtonBar from 'sentry/components/buttonBar'; import Link from 'sentry/components/links/link'; import {IconCheckmark} from 'sentry/icons'; import {t} from 'sentry/locale'; import pulsingIndicatorStyles from 'sentry/styles/pulsingIndicator'; import {space} from 'sentry/styles/space'; import type {OnboardingRecentCreatedProject} from 'sentry/types/onboarding'; import type {Organization} from 'sentry/types/organization'; import {trackAnalytics} from 'sentry/utils/analytics'; import testableTransition from 'sentry/utils/testableTransition'; import CreateSampleEventButton from 'sentry/views/onboarding/createSampleEventButton'; import GenericFooter from './genericFooter'; interface FirstEventFooterProps { isLast: boolean; onClickSetupLater: () => void; organization: Organization; project: OnboardingRecentCreatedProject; } export default function FirstEventFooter({ organization, project, onClickSetupLater, isLast, }: FirstEventFooterProps) { const source = 'targeted_onboarding_first_event_footer'; const getSecondaryCta = useCallback(() => { // if hasn't sent first event, allow skiping. // if last, no secondary cta if (!project?.firstError && !isLast) { return ; } return null; }, [project?.firstError, isLast, onClickSetupLater]); const getPrimaryCta = useCallback(() => { // if hasn't sent first event, allow creation of sample error if (!project?.firstError) { return ( {t('View Sample Error')} ); } return ( {t('Take me to my error')} ); }, [project, organization.slug]); return ( { trackAnalytics('growth.onboarding_clicked_skip', { organization, source, }); }} to={`/organizations/${organization.slug}/issues/?referrer=onboarding-first-event-footer-skip`} > {t('Skip Onboarding')} {project?.firstError ? ( ) : ( )} {project?.firstError ? t('Error Received') : t('Waiting for error')} {getSecondaryCta()} {getPrimaryCta()} ); } const OnboardingButtonBar = styled(ButtonBar)` margin: ${space(2)} ${space(4)}; justify-self: end; margin-left: auto; `; const AnimatedText = styled(motion.div, { shouldForwardProp: prop => prop !== 'errorReceived', })<{errorReceived: boolean}>` margin-left: ${space(1)}; color: ${p => (p.errorReceived ? p.theme.successText : p.theme.pink400)}; `; const indicatorAnimation: Variants = { initial: {opacity: 0, y: -10}, animate: {opacity: 1, y: 0}, exit: {opacity: 0, y: 10}, }; AnimatedText.defaultProps = { variants: indicatorAnimation, transition: testableTransition(), }; const WaitingIndicator = styled(motion.div)` ${pulsingIndicatorStyles}; background-color: ${p => p.theme.pink300}; `; WaitingIndicator.defaultProps = { variants: indicatorAnimation, transition: testableTransition(), }; const StatusWrapper = styled(motion.div)` display: flex; align-items: center; font-size: ${p => p.theme.fontSizeMedium}; justify-content: center; @media (max-width: ${p => p.theme.breakpoints.small}) { display: none; } `; StatusWrapper.defaultProps = { initial: 'initial', animate: 'animate', exit: 'exit', variants: { initial: {opacity: 0, y: -10}, animate: { opacity: 1, y: 0, transition: testableTransition({when: 'beforeChildren', staggerChildren: 0.35}), }, exit: {opacity: 0, y: 10}, }, }; const SkipOnboardingLink = styled(Link)` margin: auto ${space(4)}; white-space: nowrap; @media (max-width: ${p => p.theme.breakpoints.small}) { display: none; } `; const GridFooter = styled(GenericFooter)` display: grid; grid-template-columns: 1fr 1fr 1fr; @media (max-width: ${p => p.theme.breakpoints.small}) { display: flex; flex-direction: row; justify-content: end; } `;