From 0da951bd1056cea0ca8029dd8004b4d7787ee858 Mon Sep 17 00:00:00 2001 From: Pawan Jain Date: Thu, 11 Apr 2024 15:53:46 +0530 Subject: [PATCH 1/7] feat: add modal on get started page for onboarding experiment v2 --- .../auth/components/QuestionnaireForm.tsx | 2 +- .../components/OnboadingExperimentModal.tsx | 123 ++++++++++++++++++ .../pages/quick-start/steps/GetStarted.tsx | 4 +- 3 files changed, 127 insertions(+), 2 deletions(-) create mode 100644 apps/web/src/pages/quick-start/components/OnboadingExperimentModal.tsx diff --git a/apps/web/src/pages/auth/components/QuestionnaireForm.tsx b/apps/web/src/pages/auth/components/QuestionnaireForm.tsx index 1b9f2404cb7..a43b972a548 100644 --- a/apps/web/src/pages/auth/components/QuestionnaireForm.tsx +++ b/apps/web/src/pages/auth/components/QuestionnaireForm.tsx @@ -66,7 +66,7 @@ export function QuestionnaireForm() { const createDto: ICreateOrganizationDto = { ...rest, name: organizationName }; const organization = await createOrganizationMutation(createDto); const organizationResponseToken = await api.post(`/v1/auth/organizations/${organization._id}/switch`, {}); - + localStorage.setItem('onboarding_modal', 'true'); setToken(organizationResponseToken); } diff --git a/apps/web/src/pages/quick-start/components/OnboadingExperimentModal.tsx b/apps/web/src/pages/quick-start/components/OnboadingExperimentModal.tsx new file mode 100644 index 00000000000..b7adfd69c70 --- /dev/null +++ b/apps/web/src/pages/quick-start/components/OnboadingExperimentModal.tsx @@ -0,0 +1,123 @@ +import { useState } from 'react'; +import { Modal, useMantineTheme, Grid } from '@mantine/core'; + +import styled from '@emotion/styled'; +import { colors, shadows, Title, Button } from '@novu/design-system'; +import { useAuthContext, useSegment } from '@novu/shared-web'; +import { useCreateOnboardingExperimentWorkflow } from '../../../api/hooks/notification-templates/useCreateOnboardingExperimentWorkflow'; + +export function OnboardingExperimentModal() { + const [opened, setOpened] = useState(true); + const theme = useMantineTheme(); + const segment = useSegment(); + const { currentOrganization } = useAuthContext(); + const { + createOnboardingExperimentWorkflow, + isLoading: IsCreateOnboardingExpWorkflowLoading, + isDisabled: isIsCreateOnboardingExpWorkflowDisabled, + } = useCreateOnboardingExperimentWorkflow(); + const handleOnClose = () => { + setOpened(true); + }; + + return ( + What would you like to do first?} + sx={{ backdropFilter: 'blur(10px)' }} + shadow={theme.colorScheme === 'dark' ? shadows.dark : shadows.medium} + radius="md" + size="lg" + onClose={handleOnClose} + centered + withCloseButton={false} + > + + + + {'Send test notification'} + {'Learn how to setup a workflow and send your first email notification.'} + { + segment.track('Buton Clicked - [Onboarding]', { + action: 'Modal - Send test notification', + experiment_id: '2024-w15-onb', + _organization: currentOrganization?._id, + }); + localStorage.removeItem('onboarding_modal'); + createOnboardingExperimentWorkflow(); + }} + > + {'Send test notification now'} + + + + + + {'Look around'} + {'Start exploring the Novu app on your own terms'} + { + segment.track('Buton Clicked - [Onboarding]', { + action: 'Modal - Get started', + experiment_id: '2024-w15-onb', + _organization: currentOrganization?._id, + }); + localStorage.removeItem('onboarding_modal'); + setOpened(false); + }} + > + {'Get started'} + + + + + + ); +} + +const ChannelCard = styled.div` + display: flex; + justify-content: space-between; + flex-direction: column; + max-width: 230px; +`; + +const TitleRow = styled.div` + display: flex; + align-items: center; + font-size: 20px; + line-height: 32px; + margin-bottom: 8px; +`; + +const Description = styled.div` + flex: auto; + font-size: 16px; + line-height: 20px; + margin-bottom: 20px; + color: ${colors.B60}; + height: 60px; +`; + +const StyledButton = styled(Button)` + width: fit-content; + outline: none; +`; diff --git a/apps/web/src/pages/quick-start/steps/GetStarted.tsx b/apps/web/src/pages/quick-start/steps/GetStarted.tsx index 74618334d5c..263db0c3d73 100644 --- a/apps/web/src/pages/quick-start/steps/GetStarted.tsx +++ b/apps/web/src/pages/quick-start/steps/GetStarted.tsx @@ -9,6 +9,7 @@ import { ChannelsConfiguration } from '../components/ChannelsConfiguration'; import { GetStartedLayout } from '../components/layout/GetStartedLayout'; import { NavButton } from '../components/NavButton'; import { getStartedSteps, OnBoardingAnalyticsEnum } from '../consts'; +import { OnboardingExperimentModal } from '../components/OnboadingExperimentModal'; const ChannelsConfigurationHolder = styled.div` display: flex; @@ -29,7 +30,7 @@ export function GetStarted() { open: boolean; channelType?: ChannelTypeEnum; }>({ open: false }); - + const isOnboardingModalEnabled = localStorage.getItem('onboarding_modal') === 'true'; const onIntegrationModalClose = () => setClickedChannel({ open: false }); useEffect(() => { @@ -60,6 +61,7 @@ export function GetStarted() { /> + {isOnboardingModalEnabled && } ); } From 73740fc13ef0a6e2846fb9ba7d3015a7a897330c Mon Sep 17 00:00:00 2001 From: Pawan Jain Date: Thu, 11 Apr 2024 16:14:29 +0530 Subject: [PATCH 2/7] fix: add segment event for modal open --- apps/web/package.json | 2 +- .../src/pages/quick-start/steps/GetStarted.tsx | 16 ++++++++++++++-- 2 files changed, 15 insertions(+), 3 deletions(-) diff --git a/apps/web/package.json b/apps/web/package.json index 84356a43fd5..99a040fd04e 100644 --- a/apps/web/package.json +++ b/apps/web/package.json @@ -3,7 +3,7 @@ "version": "0.24.1", "private": true, "scripts": { - "start": "pnpm panda --watch & cross-env PORT=4200 react-app-rewired start", + "start": "pnpm panda --watch & cross-env NODE_OPTIONS=--max_old_space_size=8192 PORT=4200 react-app-rewired start", "prebuild": "rimraf build", "build": "pnpm panda && cross-env NODE_OPTIONS=--max_old_space_size=4096 GENERATE_SOURCEMAP=false react-app-rewired --max_old_space_size=4096 build", "precommit": "lint-staged", diff --git a/apps/web/src/pages/quick-start/steps/GetStarted.tsx b/apps/web/src/pages/quick-start/steps/GetStarted.tsx index 263db0c3d73..e7993505f8a 100644 --- a/apps/web/src/pages/quick-start/steps/GetStarted.tsx +++ b/apps/web/src/pages/quick-start/steps/GetStarted.tsx @@ -10,6 +10,7 @@ import { GetStartedLayout } from '../components/layout/GetStartedLayout'; import { NavButton } from '../components/NavButton'; import { getStartedSteps, OnBoardingAnalyticsEnum } from '../consts'; import { OnboardingExperimentModal } from '../components/OnboadingExperimentModal'; +import { ENV, IS_DOCKER_HOSTED, useAuthContext } from '@novu/shared-web'; const ChannelsConfigurationHolder = styled.div` display: flex; @@ -26,16 +27,27 @@ const ChannelsConfigurationHolder = styled.div` export function GetStarted() { const segment = useSegment(); + const { currentOrganization } = useAuthContext(); const [clickedChannel, setClickedChannel] = useState<{ open: boolean; channelType?: ChannelTypeEnum; }>({ open: false }); - const isOnboardingModalEnabled = localStorage.getItem('onboarding_modal') === 'true'; + + const isNovuProd = !IS_DOCKER_HOSTED && ENV === 'production'; + // open modal only for prod users + const isOnboardingModalEnabled = isNovuProd && localStorage.getItem('onboarding_modal') === 'true'; + const onIntegrationModalClose = () => setClickedChannel({ open: false }); useEffect(() => { segment.track(OnBoardingAnalyticsEnum.CONFIGURE_PROVIDER_VISIT); - }, [segment]); + if (isOnboardingModalEnabled) { + segment.track('Welcome modal open - [Onboarding]', { + experiment_id: '2024-w15-onb', + _organization: currentOrganization?._id, + }); + } + }, [currentOrganization?._id, isOnboardingModalEnabled, segment]); function handleOnClick() { segment.track(OnBoardingAnalyticsEnum.CONFIGURE_PROVIDER_NAVIGATION_NEXT_PAGE_CLICK); From 5bb71259492cbb747f065e95dea5890e0443207d Mon Sep 17 00:00:00 2001 From: Pawan Jain Date: Thu, 11 Apr 2024 16:39:44 +0530 Subject: [PATCH 3/7] fix: cspell check error --- ...boadingExperimentModal.tsx => OnboardingExperimentModal.tsx} | 2 +- apps/web/src/pages/quick-start/steps/GetStarted.tsx | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) rename apps/web/src/pages/quick-start/components/{OnboadingExperimentModal.tsx => OnboardingExperimentModal.tsx} (98%) diff --git a/apps/web/src/pages/quick-start/components/OnboadingExperimentModal.tsx b/apps/web/src/pages/quick-start/components/OnboardingExperimentModal.tsx similarity index 98% rename from apps/web/src/pages/quick-start/components/OnboadingExperimentModal.tsx rename to apps/web/src/pages/quick-start/components/OnboardingExperimentModal.tsx index b7adfd69c70..d1b7a47ac74 100644 --- a/apps/web/src/pages/quick-start/components/OnboadingExperimentModal.tsx +++ b/apps/web/src/pages/quick-start/components/OnboardingExperimentModal.tsx @@ -55,7 +55,7 @@ export function OnboardingExperimentModal() { pulse={true} variant={'gradient'} onClick={async () => { - segment.track('Buton Clicked - [Onboarding]', { + segment.track('Button Clicked - [Onboarding]', { action: 'Modal - Send test notification', experiment_id: '2024-w15-onb', _organization: currentOrganization?._id, diff --git a/apps/web/src/pages/quick-start/steps/GetStarted.tsx b/apps/web/src/pages/quick-start/steps/GetStarted.tsx index e7993505f8a..2fb0e14b6a4 100644 --- a/apps/web/src/pages/quick-start/steps/GetStarted.tsx +++ b/apps/web/src/pages/quick-start/steps/GetStarted.tsx @@ -9,7 +9,7 @@ import { ChannelsConfiguration } from '../components/ChannelsConfiguration'; import { GetStartedLayout } from '../components/layout/GetStartedLayout'; import { NavButton } from '../components/NavButton'; import { getStartedSteps, OnBoardingAnalyticsEnum } from '../consts'; -import { OnboardingExperimentModal } from '../components/OnboadingExperimentModal'; +import { OnboardingExperimentModal } from '../components/OnboardingExperimentModal'; import { ENV, IS_DOCKER_HOSTED, useAuthContext } from '@novu/shared-web'; const ChannelsConfigurationHolder = styled.div` From 52cf53cde77e1ee7190efb602373e02dbfa8a1f5 Mon Sep 17 00:00:00 2001 From: Pawan Jain Date: Thu, 11 Apr 2024 17:02:33 +0530 Subject: [PATCH 4/7] feat: change button text --- apps/web/src/pages/templates/workflow/WorkflowEditor.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/apps/web/src/pages/templates/workflow/WorkflowEditor.tsx b/apps/web/src/pages/templates/workflow/WorkflowEditor.tsx index fcd04da28a3..fcbb4ae8234 100644 --- a/apps/web/src/pages/templates/workflow/WorkflowEditor.tsx +++ b/apps/web/src/pages/templates/workflow/WorkflowEditor.tsx @@ -293,7 +293,7 @@ const WorkflowEditor = () => { }} data-test-id="get-snippet-btn" > - {tagsIncludesOnboarding ? 'Test Notification Now' : 'Get Snippet'} + {tagsIncludesOnboarding ? 'Trigger Notification' : 'Get Snippet'} From e4172c77b1cf07454ca833b551dee0d7e52bd27f Mon Sep 17 00:00:00 2001 From: Pawan Jain Date: Thu, 11 Apr 2024 17:37:45 +0530 Subject: [PATCH 5/7] fix: spell check error --- .../pages/quick-start/components/OnboardingExperimentModal.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/apps/web/src/pages/quick-start/components/OnboardingExperimentModal.tsx b/apps/web/src/pages/quick-start/components/OnboardingExperimentModal.tsx index d1b7a47ac74..8301c06ef53 100644 --- a/apps/web/src/pages/quick-start/components/OnboardingExperimentModal.tsx +++ b/apps/web/src/pages/quick-start/components/OnboardingExperimentModal.tsx @@ -75,7 +75,7 @@ export function OnboardingExperimentModal() { { - segment.track('Buton Clicked - [Onboarding]', { + segment.track('Button Clicked - [Onboarding]', { action: 'Modal - Get started', experiment_id: '2024-w15-onb', _organization: currentOrganization?._id, From d1c7fc8a96ef67eaf560726d550e131c49713596 Mon Sep 17 00:00:00 2001 From: Pawan Jain Date: Fri, 12 Apr 2024 17:11:54 +0530 Subject: [PATCH 6/7] fix: add const for local storage key --- apps/web/package.json | 2 +- .../web/src/constants/experimentsConstants.ts | 1 + .../auth/components/QuestionnaireForm.tsx | 3 +- .../components/OnboardingExperimentModal.tsx | 26 +++++++++-------- .../pages/quick-start/steps/GetStarted.tsx | 3 +- .../components/TriggerSnippetTabs.tsx | 29 +++++++++---------- .../templates/workflow/WorkflowEditor.tsx | 2 +- 7 files changed, 35 insertions(+), 31 deletions(-) create mode 100644 apps/web/src/constants/experimentsConstants.ts diff --git a/apps/web/package.json b/apps/web/package.json index 99a040fd04e..84356a43fd5 100644 --- a/apps/web/package.json +++ b/apps/web/package.json @@ -3,7 +3,7 @@ "version": "0.24.1", "private": true, "scripts": { - "start": "pnpm panda --watch & cross-env NODE_OPTIONS=--max_old_space_size=8192 PORT=4200 react-app-rewired start", + "start": "pnpm panda --watch & cross-env PORT=4200 react-app-rewired start", "prebuild": "rimraf build", "build": "pnpm panda && cross-env NODE_OPTIONS=--max_old_space_size=4096 GENERATE_SOURCEMAP=false react-app-rewired --max_old_space_size=4096 build", "precommit": "lint-staged", diff --git a/apps/web/src/constants/experimentsConstants.ts b/apps/web/src/constants/experimentsConstants.ts new file mode 100644 index 00000000000..2720076d019 --- /dev/null +++ b/apps/web/src/constants/experimentsConstants.ts @@ -0,0 +1 @@ +export const OnboardingExperimentV2ModalKey = 'nv_onboarding_modal'; diff --git a/apps/web/src/pages/auth/components/QuestionnaireForm.tsx b/apps/web/src/pages/auth/components/QuestionnaireForm.tsx index a43b972a548..d698ba91821 100644 --- a/apps/web/src/pages/auth/components/QuestionnaireForm.tsx +++ b/apps/web/src/pages/auth/components/QuestionnaireForm.tsx @@ -26,6 +26,7 @@ import { useVercelIntegration, useVercelParams } from '../../../hooks'; import { ROUTES } from '../../../constants/routes.enum'; import { DynamicCheckBox } from './dynamic-checkbox/DynamicCheckBox'; import styled from '@emotion/styled/macro'; +import { OnboardingExperimentV2ModalKey } from '../../../constants/experimentsConstants'; export function QuestionnaireForm() { const [loading, setLoading] = useState(); @@ -66,7 +67,7 @@ export function QuestionnaireForm() { const createDto: ICreateOrganizationDto = { ...rest, name: organizationName }; const organization = await createOrganizationMutation(createDto); const organizationResponseToken = await api.post(`/v1/auth/organizations/${organization._id}/switch`, {}); - localStorage.setItem('onboarding_modal', 'true'); + localStorage.setItem(OnboardingExperimentV2ModalKey, 'true'); setToken(organizationResponseToken); } diff --git a/apps/web/src/pages/quick-start/components/OnboardingExperimentModal.tsx b/apps/web/src/pages/quick-start/components/OnboardingExperimentModal.tsx index 8301c06ef53..4aaf6e36e00 100644 --- a/apps/web/src/pages/quick-start/components/OnboardingExperimentModal.tsx +++ b/apps/web/src/pages/quick-start/components/OnboardingExperimentModal.tsx @@ -5,6 +5,8 @@ import styled from '@emotion/styled'; import { colors, shadows, Title, Button } from '@novu/design-system'; import { useAuthContext, useSegment } from '@novu/shared-web'; import { useCreateOnboardingExperimentWorkflow } from '../../../api/hooks/notification-templates/useCreateOnboardingExperimentWorkflow'; +import { OnboardingExperimentV2ModalKey } from '../../../constants/experimentsConstants'; +import { OnBoardingAnalyticsEnum } from '../consts'; export function OnboardingExperimentModal() { const [opened, setOpened] = useState(true); @@ -47,44 +49,44 @@ export function OnboardingExperimentModal() { - {'Send test notification'} - {'Learn how to setup a workflow and send your first email notification.'} + Send test notification + Learn how to setup a workflow and send your first email notification. { - segment.track('Button Clicked - [Onboarding]', { + segment.track(OnBoardingAnalyticsEnum.ONBOARDING_EXPERIMENT_TEST_NOTIFICATION, { action: 'Modal - Send test notification', experiment_id: '2024-w15-onb', _organization: currentOrganization?._id, }); - localStorage.removeItem('onboarding_modal'); + localStorage.removeItem(OnboardingExperimentV2ModalKey); createOnboardingExperimentWorkflow(); }} > - {'Send test notification now'} + Send test notification now - {'Look around'} - {'Start exploring the Novu app on your own terms'} + Look around + Start exploring the Novu app on your own terms { - segment.track('Button Clicked - [Onboarding]', { + segment.track(OnBoardingAnalyticsEnum.ONBOARDING_EXPERIMENT_TEST_NOTIFICATION, { action: 'Modal - Get started', experiment_id: '2024-w15-onb', _organization: currentOrganization?._id, }); - localStorage.removeItem('onboarding_modal'); + localStorage.removeItem(OnboardingExperimentV2ModalKey); setOpened(false); }} > - {'Get started'} + Get started diff --git a/apps/web/src/pages/quick-start/steps/GetStarted.tsx b/apps/web/src/pages/quick-start/steps/GetStarted.tsx index 2fb0e14b6a4..78cffc5ae91 100644 --- a/apps/web/src/pages/quick-start/steps/GetStarted.tsx +++ b/apps/web/src/pages/quick-start/steps/GetStarted.tsx @@ -11,6 +11,7 @@ import { NavButton } from '../components/NavButton'; import { getStartedSteps, OnBoardingAnalyticsEnum } from '../consts'; import { OnboardingExperimentModal } from '../components/OnboardingExperimentModal'; import { ENV, IS_DOCKER_HOSTED, useAuthContext } from '@novu/shared-web'; +import { OnboardingExperimentV2ModalKey } from '../../../constants/experimentsConstants'; const ChannelsConfigurationHolder = styled.div` display: flex; @@ -35,7 +36,7 @@ export function GetStarted() { const isNovuProd = !IS_DOCKER_HOSTED && ENV === 'production'; // open modal only for prod users - const isOnboardingModalEnabled = isNovuProd && localStorage.getItem('onboarding_modal') === 'true'; + const isOnboardingModalEnabled = isNovuProd && localStorage.getItem(OnboardingExperimentV2ModalKey) === 'true'; const onIntegrationModalClose = () => setClickedChannel({ open: false }); diff --git a/apps/web/src/pages/templates/components/TriggerSnippetTabs.tsx b/apps/web/src/pages/templates/components/TriggerSnippetTabs.tsx index 19c7e846d5a..e7d31ec13f7 100644 --- a/apps/web/src/pages/templates/components/TriggerSnippetTabs.tsx +++ b/apps/web/src/pages/templates/components/TriggerSnippetTabs.tsx @@ -66,8 +66,7 @@ novu.trigger('${identifier}', ${JSON.stringify( 2 ) .replace(/"([^"]+)":/g, '$1:') - .replace(/"/g, "'") - .replaceAll('\n', '\n ')}); + .replace(/"/g, "'")}); `; return ( @@ -85,19 +84,19 @@ export const getCurlTriggerSnippet = ( snippet?: Record ) => { const curlSnippet = `curl --location --request POST '${API_ROOT}/v1/events/trigger' \\ - --header 'Authorization: ApiKey ' \\ - --header 'Content-Type: application/json' \\ - --data-raw '${JSON.stringify( - { - name: identifier, - to, - payload, - overrides, - ...snippet, - }, - null, - 2 - ).replaceAll('\n', '\n ')}' +--header 'Authorization: ApiKey ' \\ +--header 'Content-Type: application/json' \\ +--data-raw '${JSON.stringify( + { + name: identifier, + to, + payload, + overrides, + ...snippet, + }, + null, + 2 + )}' `; return ( diff --git a/apps/web/src/pages/templates/workflow/WorkflowEditor.tsx b/apps/web/src/pages/templates/workflow/WorkflowEditor.tsx index fcbb4ae8234..61b3ac9c551 100644 --- a/apps/web/src/pages/templates/workflow/WorkflowEditor.tsx +++ b/apps/web/src/pages/templates/workflow/WorkflowEditor.tsx @@ -293,7 +293,7 @@ const WorkflowEditor = () => { }} data-test-id="get-snippet-btn" > - {tagsIncludesOnboarding ? 'Trigger Notification' : 'Get Snippet'} + Trigger Notification From d0de31ed10634b7a191ffee284307345951a2002 Mon Sep 17 00:00:00 2001 From: Pawan Jain Date: Fri, 12 Apr 2024 19:26:03 +0530 Subject: [PATCH 7/7] fix: remove prod only condition --- apps/web/src/pages/quick-start/steps/GetStarted.tsx | 6 ++---- 1 file changed, 2 insertions(+), 4 deletions(-) diff --git a/apps/web/src/pages/quick-start/steps/GetStarted.tsx b/apps/web/src/pages/quick-start/steps/GetStarted.tsx index 78cffc5ae91..4a01f78f9b0 100644 --- a/apps/web/src/pages/quick-start/steps/GetStarted.tsx +++ b/apps/web/src/pages/quick-start/steps/GetStarted.tsx @@ -10,7 +10,7 @@ import { GetStartedLayout } from '../components/layout/GetStartedLayout'; import { NavButton } from '../components/NavButton'; import { getStartedSteps, OnBoardingAnalyticsEnum } from '../consts'; import { OnboardingExperimentModal } from '../components/OnboardingExperimentModal'; -import { ENV, IS_DOCKER_HOSTED, useAuthContext } from '@novu/shared-web'; +import { useAuthContext } from '@novu/shared-web'; import { OnboardingExperimentV2ModalKey } from '../../../constants/experimentsConstants'; const ChannelsConfigurationHolder = styled.div` @@ -34,9 +34,7 @@ export function GetStarted() { channelType?: ChannelTypeEnum; }>({ open: false }); - const isNovuProd = !IS_DOCKER_HOSTED && ENV === 'production'; - // open modal only for prod users - const isOnboardingModalEnabled = isNovuProd && localStorage.getItem(OnboardingExperimentV2ModalKey) === 'true'; + const isOnboardingModalEnabled = localStorage.getItem(OnboardingExperimentV2ModalKey) === 'true'; const onIntegrationModalClose = () => setClickedChannel({ open: false });