123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242 |
- import {Fragment, useCallback, useContext, useEffect} from 'react';
- import styled from '@emotion/styled';
- import type {MotionProps} from 'framer-motion';
- import {motion} from 'framer-motion';
- import OnboardingInstall from 'sentry-images/spot/onboarding-install.svg';
- import OnboardingSetup from 'sentry-images/spot/onboarding-setup.svg';
- import {openInviteMembersModal} from 'sentry/actionCreators/modal';
- import {Button} from 'sentry/components/button';
- import Link from 'sentry/components/links/link';
- import {OnboardingContext} from 'sentry/components/onboarding/onboardingContext';
- import {t, tct} from 'sentry/locale';
- import {space} from 'sentry/styles/space';
- import {trackAnalytics} from 'sentry/utils/analytics';
- import testableTransition from 'sentry/utils/testableTransition';
- import useOrganization from 'sentry/utils/useOrganization';
- import FallingError from 'sentry/views/onboarding/components/fallingError';
- import WelcomeBackground from 'sentry/views/onboarding/components/welcomeBackground';
- import type {StepProps} from './types';
- const fadeAway: MotionProps = {
- variants: {
- initial: {opacity: 0},
- animate: {opacity: 1, filter: 'blur(0px)'},
- exit: {opacity: 0, filter: 'blur(1px)'},
- },
- transition: testableTransition({duration: 0.8}),
- };
- type TextWrapperProps = {
- cta: React.ReactNode;
- src: string;
- subText: React.ReactNode;
- title: React.ReactNode;
- };
- function InnerAction({title, subText, cta, src}: TextWrapperProps) {
- return (
- <Fragment>
- <ActionImage src={src} />
- <TextWrapper>
- <ActionTitle>{title}</ActionTitle>
- <SubText>{subText}</SubText>
- </TextWrapper>
- <ButtonWrapper>{cta}</ButtonWrapper>
- </Fragment>
- );
- }
- function TargetedOnboardingWelcome(props: StepProps) {
- const organization = useOrganization();
- const onboardingContext = useContext(OnboardingContext);
- const source = 'targeted_onboarding';
- useEffect(() => {
- trackAnalytics('growth.onboarding_start_onboarding', {
- organization,
- source,
- });
- if (onboardingContext.data.selectedSDK) {
- // At this point the selectedSDK shall be undefined but just in case, cleaning this up here too
- onboardingContext.setData({...onboardingContext.data, selectedSDK: undefined});
- }
- }, [organization, onboardingContext]);
- const handleComplete = useCallback(() => {
- trackAnalytics('growth.onboarding_clicked_instrument_app', {
- organization,
- source,
- });
- props.onComplete();
- }, [organization, source, props]);
- const handleSkipOnboarding = useCallback(() => {
- trackAnalytics('growth.onboarding_clicked_skip', {
- organization,
- source,
- });
- }, [organization, source]);
- return (
- <FallingError>
- {({fallingError, fallCount, isFalling}) => (
- <Wrapper>
- <WelcomeBackground />
- <motion.h1 {...fadeAway} style={{marginBottom: space(0.5)}}>
- {t('Welcome to Sentry')}
- </motion.h1>
- <SubHeaderText style={{marginBottom: space(4)}} {...fadeAway}>
- {t(
- 'Your code is probably broken. Maybe not. Find out for sure. Get started below.'
- )}
- </SubHeaderText>
- <ActionItem {...fadeAway}>
- <InnerAction
- title={t('Install Sentry')}
- subText={t(
- 'Select your languages or frameworks and install the SDKs to start tracking issues'
- )}
- src={OnboardingInstall}
- cta={
- <Fragment>
- <ButtonWithFill onClick={handleComplete} priority="primary">
- {t('Start')}
- </ButtonWithFill>
- {(fallCount === 0 || isFalling) && (
- <PositionedFallingError>{fallingError}</PositionedFallingError>
- )}
- </Fragment>
- }
- />
- </ActionItem>
- <ActionItem {...fadeAway}>
- <InnerAction
- title={t('Set up my team')}
- subText={tct(
- 'Invite [friends] coworkers. You shouldn’t have to fix what you didn’t break',
- {friends: <Strike>{t('friends')}</Strike>}
- )}
- src={OnboardingSetup}
- cta={
- <ButtonWithFill
- onClick={() => {
- openInviteMembersModal({source});
- }}
- priority="primary"
- >
- {t('Invite Team')}
- </ButtonWithFill>
- }
- />
- </ActionItem>
- <motion.p style={{margin: 0}} {...fadeAway}>
- {t("Gee, I've used Sentry before.")}
- <br />
- <Link
- onClick={handleSkipOnboarding}
- to={`/organizations/${organization.slug}/issues/?referrer=onboarding-welcome-skip`}
- >
- {t('Skip onboarding.')}
- </Link>
- </motion.p>
- </Wrapper>
- )}
- </FallingError>
- );
- }
- export default TargetedOnboardingWelcome;
- const PositionedFallingError = styled('span')`
- display: block;
- position: absolute;
- right: 0px;
- top: 30px;
- `;
- const Wrapper = styled(motion.div)`
- position: relative;
- margin-top: auto;
- margin-bottom: auto;
- max-width: 400px;
- display: flex;
- flex-direction: column;
- align-items: center;
- text-align: center;
- margin-left: auto;
- margin-right: auto;
- h1 {
- font-size: 42px;
- }
- `;
- const ActionItem = styled(motion.div)`
- min-height: 120px;
- border-radius: ${space(0.5)};
- padding: ${space(2)};
- margin-bottom: ${space(2)};
- justify-content: space-around;
- border: 1px solid ${p => p.theme.gray200};
- @media (min-width: ${p => p.theme.breakpoints.small}) {
- display: grid;
- grid-template-columns: 125px auto 125px;
- width: 680px;
- align-items: center;
- }
- @media (max-width: ${p => p.theme.breakpoints.small}) {
- display: flex;
- flex-direction: column;
- }
- `;
- const TextWrapper = styled('div')`
- text-align: left;
- margin: auto ${space(3)};
- min-height: 70px;
- @media (max-width: ${p => p.theme.breakpoints.small}) {
- text-align: center;
- margin: ${space(1)} ${space(1)};
- margin-top: ${space(3)};
- }
- `;
- const Strike = styled('span')`
- text-decoration: line-through;
- `;
- const ActionTitle = styled('h5')`
- font-weight: 900;
- margin: 0 0 ${space(0.5)};
- color: ${p => p.theme.gray400};
- `;
- const SubText = styled('span')`
- font-weight: 400;
- color: ${p => p.theme.gray400};
- `;
- const SubHeaderText = styled(motion.h6)`
- color: ${p => p.theme.gray300};
- `;
- const ButtonWrapper = styled('div')`
- margin: ${space(1)};
- position: relative;
- `;
- const ActionImage = styled('img')`
- height: 100px;
- `;
- const ButtonWithFill = styled(Button)`
- width: 100%;
- position: relative;
- z-index: 1;
- `;
|