import {useCallback, useEffect, useRef, useState} from 'react'; import {browserHistory, RouteComponentProps} from 'react-router'; import styled from '@emotion/styled'; import * as Sentry from '@sentry/react'; import {AnimatePresence, motion, MotionProps, useAnimation} from 'framer-motion'; import {removeProject} from 'sentry/actionCreators/projects'; import {Button, ButtonProps} from 'sentry/components/button'; import Hook from 'sentry/components/hook'; import Link from 'sentry/components/links/link'; import LogoSentry from 'sentry/components/logoSentry'; import SentryDocumentTitle from 'sentry/components/sentryDocumentTitle'; import {IconArrow} from 'sentry/icons'; import {t} from 'sentry/locale'; import {space} from 'sentry/styles/space'; import {Organization, Project} from 'sentry/types'; import trackAdvancedAnalyticsEvent from 'sentry/utils/analytics/trackAdvancedAnalyticsEvent'; import Redirect from 'sentry/utils/redirect'; import testableTransition from 'sentry/utils/testableTransition'; import useApi from 'sentry/utils/useApi'; import {normalizeUrl} from 'sentry/utils/withDomainRequired'; import withOrganization from 'sentry/utils/withOrganization'; import withProjects from 'sentry/utils/withProjects'; import PageCorners from 'sentry/views/onboarding/components/pageCorners'; import Stepper from './components/stepper'; import OnboardingPlatform from './deprecatedPlatform'; import {PlatformSelection} from './platformSelection'; import SetupDocs from './setupDocs'; import {StepDescriptor} from './types'; import {usePersistedOnboardingState} from './utils'; import TargetedOnboardingWelcome from './welcome'; type RouteParams = { step: string; }; type Props = RouteComponentProps & { organization: Organization; projects: Project[]; }; function getOrganizationOnboardingSteps(singleSelectPlatform: boolean): StepDescriptor[] { return [ { id: 'welcome', title: t('Welcome'), Component: TargetedOnboardingWelcome, cornerVariant: 'top-right', }, { ...(singleSelectPlatform ? { id: 'select-platform', title: t('Select platform'), Component: PlatformSelection, hasFooter: true, cornerVariant: 'top-left', } : { id: 'select-platform', title: t('Select platforms'), Component: OnboardingPlatform, hasFooter: true, cornerVariant: 'top-left', }), }, { id: 'setup-docs', title: t('Install the Sentry SDK'), Component: SetupDocs, hasFooter: true, cornerVariant: 'top-left', }, ]; } function Onboarding(props: Props) { const api = useApi(); const { organization, params: {step: stepId}, } = props; const cornerVariantTimeoutRed = useRef(undefined); const [clientState, setClientState] = usePersistedOnboardingState(); useEffect(() => { return () => { window.clearTimeout(cornerVariantTimeoutRed.current); }; }, []); const heartbeatFooter = !!props.organization?.features.includes( 'onboarding-heartbeat-footer' ); const singleSelectPlatform = !!props.organization?.features.includes( 'onboarding-remove-multiselect-platform' ); const projectDeletionOnBackClick = !!props.organization?.features.includes( 'onboarding-project-deletion-on-back-click' ); const onboardingSteps = getOrganizationOnboardingSteps(singleSelectPlatform); const stepObj = onboardingSteps.find(({id}) => stepId === id); const stepIndex = onboardingSteps.findIndex(({id}) => stepId === id); const cornerVariantControl = useAnimation(); const updateCornerVariant = () => { // TODO: find better way to delay the corner animation window.clearTimeout(cornerVariantTimeoutRed.current); cornerVariantTimeoutRed.current = window.setTimeout( () => cornerVariantControl.start(stepIndex === 0 ? 'top-right' : 'top-left'), 1000 ); }; useEffect(updateCornerVariant, [stepIndex, cornerVariantControl]); // Called onExitComplete const [containerHasFooter, setContainerHasFooter] = useState(false); const updateAnimationState = () => { if (!stepObj) { return; } setContainerHasFooter(stepObj.hasFooter ?? false); }; const goToStep = (step: StepDescriptor) => { if (!stepObj) { return; } if (step.cornerVariant !== stepObj.cornerVariant) { cornerVariantControl.start('none'); } browserHistory.push(normalizeUrl(`/onboarding/${organization.slug}/${step.id}/`)); }; const goNextStep = (step: StepDescriptor) => { const currentStepIndex = onboardingSteps.findIndex(s => s.id === step.id); const nextStep = onboardingSteps[currentStepIndex + 1]; if (step.cornerVariant !== nextStep.cornerVariant) { cornerVariantControl.start('none'); } browserHistory.push(normalizeUrl(`/onboarding/${organization.slug}/${nextStep.id}/`)); }; const handleGoBack = () => { if (!stepObj) { return; } const previousStep = onboardingSteps[stepIndex - 1]; if (!previousStep) { return; } // The user is going back to select a new platform, // so we silently delete the last created project if (projectDeletionOnBackClick && stepIndex === onboardingSteps.length - 1) { const selectedPlatforms = clientState?.selectedPlatforms || []; const platformToProjectIdMap = clientState?.platformToProjectIdMap || {}; const selectedProjectSlugs = selectedPlatforms .map(platform => platformToProjectIdMap[platform]) .filter((slug): slug is string => slug !== undefined); removeProject(api, organization.slug, selectedProjectSlugs[0]); } if (stepObj.cornerVariant !== previousStep.cornerVariant) { cornerVariantControl.start('none'); } trackAdvancedAnalyticsEvent('heartbeat.onboarding_back_button_clicked', { organization, from: onboardingSteps[stepIndex].id, to: previousStep.id, }); browserHistory.replace( normalizeUrl(`/onboarding/${organization.slug}/${previousStep.id}/`) ); }; const genSkipOnboardingLink = () => { const source = `targeted-onboarding-${stepId}`; return ( { trackAdvancedAnalyticsEvent('growth.onboarding_clicked_skip', { organization, source, }); if (clientState) { setClientState({ ...clientState, state: 'skipped', }); } }} to={normalizeUrl( `/organizations/${organization.slug}/issues/?referrer=onboarding-skip` )} > {t('Skip Onboarding')} ); }; const jumpToSetupProject = useCallback(() => { const nextStep = onboardingSteps.find(({id}) => id === 'setup-docs'); if (!nextStep) { Sentry.captureMessage( 'Missing step in onboarding: `setup-docs` when trying to jump there' ); return; } browserHistory.push(normalizeUrl(`/onboarding/${organization.slug}/${nextStep.id}/`)); }, [onboardingSteps, organization]); if (!stepObj || stepIndex === -1) { return ( ); } return (
{stepIndex !== -1 && ( goToStep(onboardingSteps[i])} /> )}
0 ? 'visible' : 'hidden'} onClick={handleGoBack} /> {stepObj.Component && ( stepObj && goNextStep(stepObj)} orgId={organization.slug} organization={props.organization} search={props.location.search} route={props.route} router={props.router} location={props.location} jumpToSetupProject={jumpToSetupProject} {...{ genSkipOnboardingLink, }} /> )}
); } const Container = styled('div')<{hasFooter: boolean; heartbeatFooter: boolean}>` flex-grow: 1; display: flex; flex-direction: column; position: relative; background: ${p => p.theme.background}; padding: ${p => p.heartbeatFooter ? `120px ${space(3)} 0 ${space(3)}` : `120px ${space(3)}`}; width: 100%; margin: 0 auto; padding-bottom: ${p => p.hasFooter && '72px'}; margin-bottom: ${p => p.hasFooter && '72px'}; `; const Header = styled('header')` background: ${p => p.theme.background}; padding-left: ${space(4)}; padding-right: ${space(4)}; position: sticky; height: 80px; align-items: center; top: 0; z-index: 100; box-shadow: 0 5px 10px rgba(0, 0, 0, 0.05); display: grid; grid-template-columns: 1fr 1fr 1fr; justify-items: stretch; `; const LogoSvg = styled(LogoSentry)` width: 130px; height: 30px; color: ${p => p.theme.textColor}; `; const OnboardingStep = styled(motion.div)` flex-grow: 1; display: flex; flex-direction: column; `; OnboardingStep.defaultProps = { initial: 'initial', animate: 'animate', exit: 'exit', variants: {animate: {}}, transition: testableTransition({ staggerChildren: 0.2, }), }; const Sidebar = styled(motion.div)` width: 850px; display: flex; flex-direction: column; align-items: center; `; Sidebar.defaultProps = { initial: 'initial', animate: 'animate', exit: 'exit', variants: {animate: {}}, transition: testableTransition({ staggerChildren: 0.2, }), }; const AdaptivePageCorners = styled(PageCorners)` --corner-scale: 1; @media (max-width: ${p => p.theme.breakpoints.small}) { --corner-scale: 0.5; } `; const StyledStepper = styled(Stepper)` justify-self: center; @media (max-width: ${p => p.theme.breakpoints.medium}) { display: none; } `; interface BackButtonProps extends Omit { animate: MotionProps['animate']; className?: string; } const Back = styled(({className, animate, ...props}: BackButtonProps) => ( ))` position: absolute; top: 40px; left: 20px; button { font-size: ${p => p.theme.fontSizeSmall}; } `; const SkipOnboardingLink = styled(Link)` margin: auto ${space(4)}; `; const UpsellWrapper = styled('div')` grid-column: 3; margin-left: auto; `; const OnboardingWrapper = styled('main')` flex-grow: 1; display: flex; flex-direction: column; `; export default withOrganization(withProjects(Onboarding));