import {useCallback, useContext, useEffect, useRef, useState} from 'react'; import {RouteComponentProps} from 'react-router'; import styled from '@emotion/styled'; import {AnimatePresence, motion, MotionProps, useAnimation} from 'framer-motion'; import {removeProject} from 'sentry/actionCreators/projects'; import {Button, ButtonProps} from 'sentry/components/button'; import Confirm, {openConfirmModal, OpenConfirmOptions} from 'sentry/components/confirm'; import Hook from 'sentry/components/hook'; import Link from 'sentry/components/links/link'; import LogoSentry from 'sentry/components/logoSentry'; import {OnboardingContext} from 'sentry/components/onboarding/onboardingContext'; import {useRecentCreatedProject} from 'sentry/components/onboarding/useRecentCreatedProject'; import SentryDocumentTitle from 'sentry/components/sentryDocumentTitle'; import categoryList from 'sentry/data/platformCategories'; import platforms from 'sentry/data/platforms'; import {IconArrow} from 'sentry/icons'; import {t} from 'sentry/locale'; import {space} from 'sentry/styles/space'; import {OnboardingSelectedSDK} from 'sentry/types'; import {defined} from 'sentry/utils'; import {trackAnalytics} from 'sentry/utils/analytics'; import handleXhrErrorResponse from 'sentry/utils/handleXhrErrorResponse'; import Redirect from 'sentry/utils/redirect'; import testableTransition from 'sentry/utils/testableTransition'; import useApi from 'sentry/utils/useApi'; import useOrganization from 'sentry/utils/useOrganization'; import {normalizeUrl} from 'sentry/utils/withDomainRequired'; import PageCorners from 'sentry/views/onboarding/components/pageCorners'; import Stepper from './components/stepper'; import {PlatformSelection} from './platformSelection'; import SetupDocs from './setupDocs'; import {StepDescriptor} from './types'; import TargetedOnboardingWelcome from './welcome'; type RouteParams = { step: string; }; type Props = RouteComponentProps; function getOrganizationOnboardingSteps(): StepDescriptor[] { return [ { id: 'welcome', title: t('Welcome'), Component: TargetedOnboardingWelcome, cornerVariant: 'top-right', }, { id: 'select-platform', title: t('Select platform'), Component: PlatformSelection, 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 = useOrganization(); const onboardingContext = useContext(OnboardingContext); const selectedSDK = onboardingContext.data.selectedSDK; const selectedProjectSlug = selectedSDK?.key; const { params: {step: stepId}, } = props; const onboardingSteps = getOrganizationOnboardingSteps(); const stepObj = onboardingSteps.find(({id}) => stepId === id); const stepIndex = onboardingSteps.findIndex(({id}) => stepId === id); const recentCreatedProject = useRecentCreatedProject({ orgSlug: organization.slug, projectSlug: onboardingSteps[stepIndex].id === 'setup-docs' ? selectedProjectSlug : undefined, }); const cornerVariantTimeoutRed = useRef(undefined); useEffect(() => { return () => { window.clearTimeout(cornerVariantTimeoutRed.current); }; }, []); useEffect(() => { if ( props.location.pathname === `/onboarding/${onboardingSteps[2].id}/` && props.location.query?.platform && onboardingContext.data.selectedSDK === undefined ) { const platformKey = Object.keys(platforms).find( key => platforms[key].id === props.location.query.platform ); const platform = platformKey ? platforms[platformKey] : undefined; // if no platform found, we redirect the user to the platform select page if (!platform) { props.router.push( normalizeUrl(`/onboarding/${organization.slug}/${onboardingSteps[1].id}/`) ); return; } const frameworkCategory = categoryList.find(category => { return category.platforms.includes(platform.id as never); })?.id ?? 'all'; onboardingContext.setData({ ...onboardingContext.data, selectedSDK: { key: props.location.query.platform, category: frameworkCategory, language: platform.language, type: platform.type, }, }); } }, [ props.location.query, props.router, onboardingContext, onboardingSteps, organization.slug, ]); const heartbeatFooter = !!organization?.features.includes( 'onboarding-heartbeat-footer' ); const projectDeletionOnBackClick = !!organization?.features.includes( 'onboarding-project-deletion-on-back-click' ); const shallProjectBeDeleted = projectDeletionOnBackClick && onboardingSteps[stepIndex].id === 'setup-docs' && recentCreatedProject && // if the project has received a first error, we don't delete it recentCreatedProject.firstError === false && // if the project has received a first transaction, we don't delete it recentCreatedProject.firstTransaction === false && // if the project has replays, we don't delete it recentCreatedProject.hasReplays === false && // if the project has sessions, we don't delete it recentCreatedProject.hasSessions === false && // if the project is older than one hour, we don't delete it recentCreatedProject.olderThanOneHour === false; 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'); } props.router.push(normalizeUrl(`/onboarding/${organization.slug}/${step.id}/`)); }; const goNextStep = useCallback( (step: StepDescriptor, platform?: OnboardingSelectedSDK) => { const currentStepIndex = onboardingSteps.findIndex(s => s.id === step.id); const nextStep = onboardingSteps[currentStepIndex + 1]; if (nextStep.id === 'setup-docs' && !platform) { return; } if (step.cornerVariant !== nextStep.cornerVariant) { cornerVariantControl.start('none'); } props.router.push(normalizeUrl(`/onboarding/${organization.slug}/${nextStep.id}/`)); }, [organization.slug, onboardingSteps, cornerVariantControl, props.router] ); const deleteRecentCreatedProject = useCallback(async () => { if (!recentCreatedProject?.slug) { return; } const newProjects = Object.keys(onboardingContext.data.projects).reduce( (acc, key) => { if ( onboardingContext.data.projects[key].slug !== onboardingContext.data.selectedSDK?.key ) { acc[key] = onboardingContext.data.projects[key]; } return acc; }, {} ); try { await removeProject({ api, orgSlug: organization.slug, projectSlug: recentCreatedProject.slug, origin: 'onboarding', }); onboardingContext.setData({ ...onboardingContext.data, projects: newProjects, }); trackAnalytics('onboarding.data_removed', { organization, date_created: recentCreatedProject.dateCreated, platform: recentCreatedProject.slug, project_id: recentCreatedProject.id, }); } catch (error) { handleXhrErrorResponse(t('Unable to delete project in onboarding'))(error); // we don't give the user any feedback regarding this error as this shall be silent } }, [api, organization, recentCreatedProject, onboardingContext]); const handleGoBack = useCallback( (goToStepIndex?: number) => { if (!stepObj) { return; } const previousStep = defined(goToStepIndex) ? onboardingSteps[goToStepIndex] : onboardingSteps[stepIndex - 1]; if (!previousStep) { return; } if (stepObj.cornerVariant !== previousStep.cornerVariant) { cornerVariantControl.start('none'); } trackAnalytics('onboarding.back_button_clicked', { organization, from: onboardingSteps[stepIndex].id, to: previousStep.id, }); // from selected platform to welcome if (onboardingSteps[stepIndex].id === 'select-platform') { onboardingContext.setData({...onboardingContext.data, selectedSDK: undefined}); props.router.replace( normalizeUrl(`/onboarding/${organization.slug}/${previousStep.id}/`) ); return; } // from setup docs to selected platform if (onboardingSteps[stepIndex].id === 'setup-docs' && shallProjectBeDeleted) { trackAnalytics('onboarding.data_removal_modal_confirm_button_clicked', { organization, platform: recentCreatedProject.slug, project_id: recentCreatedProject.id, }); deleteRecentCreatedProject(); } props.router.replace( normalizeUrl(`/onboarding/${organization.slug}/${previousStep.id}/`) ); }, [ stepObj, stepIndex, onboardingSteps, organization, cornerVariantControl, props.router, onboardingContext, shallProjectBeDeleted, deleteRecentCreatedProject, recentCreatedProject, ] ); const genSkipOnboardingLink = () => { const source = `targeted-onboarding-${stepId}`; return ( { trackAnalytics('growth.onboarding_clicked_skip', { organization, source, }); onboardingContext.setData({...onboardingContext.data, selectedSDK: undefined}); }} to={normalizeUrl( `/organizations/${organization.slug}/issues/?referrer=onboarding-skip` )} > {t('Skip Onboarding')} ); }; if (!stepObj || stepIndex === -1) { return ( ); } const goBackDeletionAlertModalProps: OpenConfirmOptions = { message: t( "Hey, just a heads up - we haven't received any data for this SDK yet and by going back all changes will be discarded. Are you sure you want to head back?" ), priority: 'danger', confirmText: t("Yes I'm sure"), onConfirm: handleGoBack, onClose: () => { if (!recentCreatedProject) { return; } trackAnalytics('onboarding.data_removal_modal_dismissed', { organization, platform: recentCreatedProject.slug, project_id: recentCreatedProject.id, }); }, onRender: () => { if (!recentCreatedProject) { return; } trackAnalytics('onboarding.data_removal_modal_rendered', { organization, platform: recentCreatedProject.slug, project_id: recentCreatedProject.id, }); }, }; return (
{stepIndex !== -1 && ( { if (i < stepIndex && shallProjectBeDeleted) { openConfirmModal({ ...goBackDeletionAlertModalProps, onConfirm: () => handleGoBack(i), }); return; } goToStep(onboardingSteps[i]); }} /> )}
0 ? 'visible' : 'hidden'} /> {stepObj.Component && ( { if (stepObj) { goNextStep(stepObj, platform); } }} orgId={organization.slug} search={props.location.search} route={props.route} router={props.router} location={props.location} recentCreatedProject={recentCreatedProject} {...{ 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 Onboarding;