fix billingCheckoutSession query param + enable redirect on workspace… (#11509)

… during onboarding



fixes : https://github.com/twentyhq/core-team-issues/issues/668
This commit is contained in:
Etienne
2025-04-10 16:47:40 +02:00
committed by GitHub
parent d69932c6c4
commit ee5aa2393d
20 changed files with 151 additions and 92 deletions

View File

@ -1,11 +1,9 @@
import { useEffect, useState } from 'react'; import { useEffect, useState } from 'react';
import { import {
createSearchParams,
matchPath, matchPath,
useLocation, useLocation,
useNavigate, useNavigate,
useParams, useParams,
useSearchParams,
} from 'react-router-dom'; } from 'react-router-dom';
import { useRecoilValue } from 'recoil'; import { useRecoilValue } from 'recoil';
@ -65,28 +63,11 @@ export const PageChangeEffect = () => {
} }
}, [location, previousLocation, executeTasksOnAnyLocationChange]); }, [location, previousLocation, executeTasksOnAnyLocationChange]);
const [searchParams] = useSearchParams();
useEffect(() => { useEffect(() => {
if (isDefined(pageChangeEffectNavigateLocation)) { if (isDefined(pageChangeEffectNavigateLocation)) {
const hasQueryParams = pageChangeEffectNavigateLocation.includes('?'); navigate(pageChangeEffectNavigateLocation);
const navigationParams = createSearchParams({
...(searchParams.get('animateModal')
? { animateModal: searchParams.get('animateModal') ?? 'false' }
: {}),
});
if (hasQueryParams) {
navigate(pageChangeEffectNavigateLocation);
} else {
navigate({
pathname: pageChangeEffectNavigateLocation,
search: navigationParams.toString(),
});
}
} }
}, [navigate, pageChangeEffectNavigateLocation, searchParams]); }, [navigate, pageChangeEffectNavigateLocation]);
useEffect(() => { useEffect(() => {
const isLeavingRecordIndexPage = !!matchPath( const isLeavingRecordIndexPage = !!matchPath(

View File

@ -3,9 +3,11 @@ import { useSearchParams } from 'react-router-dom';
import { useIsLogged } from '@/auth/hooks/useIsLogged'; import { useIsLogged } from '@/auth/hooks/useIsLogged';
import { useVerifyLogin } from '@/auth/hooks/useVerifyLogin'; import { useVerifyLogin } from '@/auth/hooks/useVerifyLogin';
import { clientConfigApiStatusState } from '@/client-config/states/clientConfigApiStatusState';
import { AppPath } from '@/types/AppPath'; import { AppPath } from '@/types/AppPath';
import { SnackBarVariant } from '@/ui/feedback/snack-bar-manager/components/SnackBar'; import { SnackBarVariant } from '@/ui/feedback/snack-bar-manager/components/SnackBar';
import { useSnackBar } from '@/ui/feedback/snack-bar-manager/hooks/useSnackBar'; import { useSnackBar } from '@/ui/feedback/snack-bar-manager/hooks/useSnackBar';
import { useRecoilValue } from 'recoil';
import { isDefined } from 'twenty-shared/utils'; import { isDefined } from 'twenty-shared/utils';
import { useNavigateApp } from '~/hooks/useNavigateApp'; import { useNavigateApp } from '~/hooks/useNavigateApp';
import { SignInUpLoading } from '~/pages/auth/SignInUpLoading'; import { SignInUpLoading } from '~/pages/auth/SignInUpLoading';
@ -20,6 +22,10 @@ export const Verify = () => {
const navigate = useNavigateApp(); const navigate = useNavigateApp();
const { verifyLoginToken } = useVerifyLogin(); const { verifyLoginToken } = useVerifyLogin();
const { isLoaded: clientConfigLoaded } = useRecoilValue(
clientConfigApiStatusState,
);
useEffect(() => { useEffect(() => {
if (isDefined(errorMessage)) { if (isDefined(errorMessage)) {
enqueueSnackBar(errorMessage, { enqueueSnackBar(errorMessage, {
@ -28,6 +34,8 @@ export const Verify = () => {
}); });
} }
if (!clientConfigLoaded) return;
if (isDefined(loginToken)) { if (isDefined(loginToken)) {
verifyLoginToken(loginToken); verifyLoginToken(loginToken);
} else if (!isLogged) { } else if (!isLogged) {
@ -35,7 +43,7 @@ export const Verify = () => {
} }
// Verify only needs to run once at mount // Verify only needs to run once at mount
// eslint-disable-next-line react-hooks/exhaustive-deps // eslint-disable-next-line react-hooks/exhaustive-deps
}, []); }, [clientConfigLoaded]);
return <SignInUpLoading />; return <SignInUpLoading />;
}; };

View File

@ -4,10 +4,12 @@ import { SnackBarVariant } from '@/ui/feedback/snack-bar-manager/components/Snac
import { useSnackBar } from '@/ui/feedback/snack-bar-manager/hooks/useSnackBar'; import { useSnackBar } from '@/ui/feedback/snack-bar-manager/hooks/useSnackBar';
import { useVerifyLogin } from '@/auth/hooks/useVerifyLogin'; import { useVerifyLogin } from '@/auth/hooks/useVerifyLogin';
import { animateModalState } from '@/auth/states/animateModalState';
import { useRedirectToWorkspaceDomain } from '@/domain-manager/hooks/useRedirectToWorkspaceDomain'; import { useRedirectToWorkspaceDomain } from '@/domain-manager/hooks/useRedirectToWorkspaceDomain';
import { useLingui } from '@lingui/react/macro'; import { useLingui } from '@lingui/react/macro';
import { useEffect, useState } from 'react'; import { useEffect, useState } from 'react';
import { useSearchParams } from 'react-router-dom'; import { useSearchParams } from 'react-router-dom';
import { useSetRecoilState } from 'recoil';
import { useNavigateApp } from '~/hooks/useNavigateApp'; import { useNavigateApp } from '~/hooks/useNavigateApp';
import { getWorkspaceUrl } from '~/utils/getWorkspaceUrl'; import { getWorkspaceUrl } from '~/utils/getWorkspaceUrl';
import { EmailVerificationSent } from '../sign-in-up/components/EmailVerificationSent'; import { EmailVerificationSent } from '../sign-in-up/components/EmailVerificationSent';
@ -19,6 +21,8 @@ export const VerifyEmailEffect = () => {
const [searchParams] = useSearchParams(); const [searchParams] = useSearchParams();
const [isError, setIsError] = useState(false); const [isError, setIsError] = useState(false);
const setAnimateModal = useSetRecoilState(animateModalState);
const email = searchParams.get('email'); const email = searchParams.get('email');
const emailVerificationToken = searchParams.get('emailVerificationToken'); const emailVerificationToken = searchParams.get('emailVerificationToken');
@ -48,9 +52,9 @@ export const VerifyEmailEffect = () => {
const workspaceUrl = getWorkspaceUrl(workspaceUrls); const workspaceUrl = getWorkspaceUrl(workspaceUrls);
if (workspaceUrl.slice(0, -1) !== window.location.origin) { if (workspaceUrl.slice(0, -1) !== window.location.origin) {
return redirectToWorkspaceDomain(workspaceUrl, AppPath.Verify, { setAnimateModal(false);
return await redirectToWorkspaceDomain(workspaceUrl, AppPath.Verify, {
loginToken: loginToken.token, loginToken: loginToken.token,
animateModal: false,
}); });
} }
verifyLoginToken(loginToken.token); verifyLoginToken(loginToken.token);

View File

@ -41,6 +41,7 @@ import { getTimeFormatFromWorkspaceTimeFormat } from '@/localization/utils/getTi
import { currentUserState } from '../states/currentUserState'; import { currentUserState } from '../states/currentUserState';
import { tokenPairState } from '../states/tokenPairState'; import { tokenPairState } from '../states/tokenPairState';
import { animateModalState } from '@/auth/states/animateModalState';
import { currentUserWorkspaceState } from '@/auth/states/currentUserWorkspaceState'; import { currentUserWorkspaceState } from '@/auth/states/currentUserWorkspaceState';
import { import {
SignInUpStep, SignInUpStep,
@ -114,6 +115,7 @@ export const useAuth = () => {
const goToRecoilSnapshot = useGotoRecoilSnapshot(); const goToRecoilSnapshot = useGotoRecoilSnapshot();
const setDateTimeFormat = useSetRecoilState(dateTimeFormatState); const setDateTimeFormat = useSetRecoilState(dateTimeFormatState);
const setAnimateModal = useSetRecoilState(animateModalState);
const [, setSearchParams] = useSearchParams(); const [, setSearchParams] = useSearchParams();
@ -420,14 +422,13 @@ export const useAuth = () => {
} }
if (isMultiWorkspaceEnabled) { if (isMultiWorkspaceEnabled) {
return redirectToWorkspaceDomain( setAnimateModal(false);
return await redirectToWorkspaceDomain(
getWorkspaceUrl(signUpResult.data.signUp.workspace.workspaceUrls), getWorkspaceUrl(signUpResult.data.signUp.workspace.workspaceUrls),
isEmailVerificationRequired ? AppPath.SignInUp : AppPath.Verify, isEmailVerificationRequired ? AppPath.SignInUp : AppPath.Verify,
{ {
...(!isEmailVerificationRequired && { ...(!isEmailVerificationRequired && {
loginToken: signUpResult.data.signUp.loginToken.token, loginToken: signUpResult.data.signUp.loginToken.token,
animateModal: false,
}), }),
email, email,
}, },
@ -445,6 +446,7 @@ export const useAuth = () => {
handleGetAuthTokensFromLoginToken, handleGetAuthTokensFromLoginToken,
setSignInUpStep, setSignInUpStep,
setSearchParams, setSearchParams,
setAnimateModal,
isEmailVerificationRequired, isEmailVerificationRequired,
redirectToWorkspaceDomain, redirectToWorkspaceDomain,
], ],

View File

@ -90,13 +90,13 @@ export const SignInUpGlobalScopeForm = () => {
variant: SnackBarVariant.Error, variant: SnackBarVariant.Error,
}); });
}, },
onCompleted: (data) => { onCompleted: async (data) => {
requestFreshCaptchaToken(); requestFreshCaptchaToken();
const response = data.checkUserExists; const response = data.checkUserExists;
if (response.__typename === 'UserExists') { if (response.__typename === 'UserExists') {
if (response.availableWorkspaces.length >= 1) { if (response.availableWorkspaces.length >= 1) {
const workspace = response.availableWorkspaces[0]; const workspace = response.availableWorkspaces[0];
return redirectToWorkspaceDomain( return await redirectToWorkspaceDomain(
getWorkspaceUrl(workspace.workspaceUrls), getWorkspaceUrl(workspace.workspaceUrls),
pathname, pathname,
{ {

View File

@ -0,0 +1,26 @@
import { urlSyncEffect } from 'recoil-sync';
import { createState } from 'twenty-ui/utilities';
export const animateModalState = createState<boolean>({
key: 'animateModalState',
defaultValue: true,
effects: [
urlSyncEffect({
itemKey: 'animateModal',
refine: (value: unknown) => {
if (typeof value === 'boolean') {
return {
type: 'success',
value: value as boolean,
warnings: [],
} as const;
}
return {
type: 'failure',
message: 'Invalid animateModalState',
path: [] as any,
} as const;
},
}),
],
});

View File

@ -8,6 +8,7 @@ export const billingCheckoutSessionState = createState<BillingCheckoutSession>({
defaultValue: BILLING_CHECKOUT_SESSION_DEFAULT_VALUE, defaultValue: BILLING_CHECKOUT_SESSION_DEFAULT_VALUE,
effects: [ effects: [
syncEffect({ syncEffect({
itemKey: 'billingCheckoutSession',
refine: (value: unknown) => { refine: (value: unknown) => {
if ( if (
typeof value === 'object' && typeof value === 'object' &&

View File

@ -0,0 +1,32 @@
import { animateModalState } from '@/auth/states/animateModalState';
import { billingCheckoutSessionState } from '@/auth/states/billingCheckoutSessionState';
import { BILLING_CHECKOUT_SESSION_DEFAULT_VALUE } from '@/billing/constants/BillingCheckoutSessionDefaultValue';
import { useRecoilCallback } from 'recoil';
export const useBuildSearchParamsFromUrlSyncedStates = () => {
const buildSearchParamsFromUrlSyncedStates = useRecoilCallback(
({ snapshot }) =>
async () => {
const animateModal = snapshot.getLoadable(animateModalState).getValue();
const billingCheckoutSession = snapshot
.getLoadable(billingCheckoutSessionState)
.getValue();
const output = {
...(billingCheckoutSession !== BILLING_CHECKOUT_SESSION_DEFAULT_VALUE
? {
billingCheckoutSession: JSON.stringify(billingCheckoutSession),
}
: {}),
...(animateModal === false ? { animateModal: 'false' } : {}),
};
return output;
},
[],
);
return {
buildSearchParamsFromUrlSyncedStates,
};
};

View File

@ -1,6 +1,6 @@
import { useRecoilValue, useSetRecoilState } from 'recoil';
import { lastAuthenticatedWorkspaceDomainState } from '@/domain-manager/states/lastAuthenticatedWorkspaceDomainState';
import { domainConfigurationState } from '@/domain-manager/states/domainConfigurationState'; import { domainConfigurationState } from '@/domain-manager/states/domainConfigurationState';
import { lastAuthenticatedWorkspaceDomainState } from '@/domain-manager/states/lastAuthenticatedWorkspaceDomainState';
import { useRecoilValue, useSetRecoilState } from 'recoil';
export const useLastAuthenticatedWorkspaceDomain = () => { export const useLastAuthenticatedWorkspaceDomain = () => {
const domainConfiguration = useRecoilValue(domainConfigurationState); const domainConfiguration = useRecoilValue(domainConfigurationState);

View File

@ -1,4 +1,5 @@
import { isMultiWorkspaceEnabledState } from '@/client-config/states/isMultiWorkspaceEnabledState'; import { isMultiWorkspaceEnabledState } from '@/client-config/states/isMultiWorkspaceEnabledState';
import { useBuildSearchParamsFromUrlSyncedStates } from '@/domain-manager/hooks/useBuildSearchParamsFromUrlSyncedStates';
import { useBuildWorkspaceUrl } from '@/domain-manager/hooks/useBuildWorkspaceUrl'; import { useBuildWorkspaceUrl } from '@/domain-manager/hooks/useBuildWorkspaceUrl';
import { useRedirect } from '@/domain-manager/hooks/useRedirect'; import { useRedirect } from '@/domain-manager/hooks/useRedirect';
import { useRecoilValue } from 'recoil'; import { useRecoilValue } from 'recoil';
@ -8,14 +9,23 @@ export const useRedirectToWorkspaceDomain = () => {
const { buildWorkspaceUrl } = useBuildWorkspaceUrl(); const { buildWorkspaceUrl } = useBuildWorkspaceUrl();
const { redirect } = useRedirect(); const { redirect } = useRedirect();
const redirectToWorkspaceDomain = ( const { buildSearchParamsFromUrlSyncedStates } =
useBuildSearchParamsFromUrlSyncedStates();
const redirectToWorkspaceDomain = async (
baseUrl: string, baseUrl: string,
pathname?: string, pathname?: string,
searchParams?: Record<string, string | boolean>, searchParams?: Record<string, string | boolean>,
target?: string, target?: string,
) => { ) => {
if (!isMultiWorkspaceEnabled) return; if (!isMultiWorkspaceEnabled) return;
redirect(buildWorkspaceUrl(baseUrl, pathname, searchParams), target); redirect(
buildWorkspaceUrl(baseUrl, pathname, {
...searchParams,
...(await buildSearchParamsFromUrlSyncedStates()),
}),
target,
);
}; };
return { return {

View File

@ -1,19 +1,23 @@
import { animateModalState } from '@/auth/states/animateModalState';
import { useRedirectToWorkspaceDomain } from '@/domain-manager/hooks/useRedirectToWorkspaceDomain'; import { useRedirectToWorkspaceDomain } from '@/domain-manager/hooks/useRedirectToWorkspaceDomain';
import { AppPath } from '@/types/AppPath'; import { AppPath } from '@/types/AppPath';
import { useSetRecoilState } from 'recoil';
import { WorkspaceUrls } from '~/generated/graphql'; import { WorkspaceUrls } from '~/generated/graphql';
import { getWorkspaceUrl } from '~/utils/getWorkspaceUrl'; import { getWorkspaceUrl } from '~/utils/getWorkspaceUrl';
export const useImpersonationRedirect = () => { export const useImpersonationRedirect = () => {
const { redirectToWorkspaceDomain } = useRedirectToWorkspaceDomain(); const { redirectToWorkspaceDomain } = useRedirectToWorkspaceDomain();
const setAnimateModal = useSetRecoilState(animateModalState);
const executeImpersonationRedirect = ( const executeImpersonationRedirect = async (
workspaceUrls: WorkspaceUrls, workspaceUrls: WorkspaceUrls,
loginToken: string, loginToken: string,
) => { ) => {
return redirectToWorkspaceDomain( setAnimateModal(false);
return await redirectToWorkspaceDomain(
getWorkspaceUrl(workspaceUrls), getWorkspaceUrl(workspaceUrls),
AppPath.Verify, AppPath.Verify,
{ loginToken, animateModal: false }, { loginToken },
); );
}; };

View File

@ -1,4 +1,5 @@
import { AuthModal } from '@/auth/components/AuthModal'; import { AuthModal } from '@/auth/components/AuthModal';
import { animateModalState } from '@/auth/states/animateModalState';
import { CommandMenuRouter } from '@/command-menu/components/CommandMenuRouter'; import { CommandMenuRouter } from '@/command-menu/components/CommandMenuRouter';
import { AppErrorBoundary } from '@/error-handler/components/AppErrorBoundary'; import { AppErrorBoundary } from '@/error-handler/components/AppErrorBoundary';
import { AppFullScreenErrorFallback } from '@/error-handler/components/AppFullScreenErrorFallback'; import { AppFullScreenErrorFallback } from '@/error-handler/components/AppFullScreenErrorFallback';
@ -17,7 +18,8 @@ import { useIsMobile } from '@/ui/utilities/responsive/hooks/useIsMobile';
import { Global, css, useTheme } from '@emotion/react'; import { Global, css, useTheme } from '@emotion/react';
import styled from '@emotion/styled'; import styled from '@emotion/styled';
import { AnimatePresence, LayoutGroup, motion } from 'framer-motion'; import { AnimatePresence, LayoutGroup, motion } from 'framer-motion';
import { Outlet, useSearchParams } from 'react-router-dom'; import { Outlet } from 'react-router-dom';
import { useRecoilValue } from 'recoil';
import { useScreenSize } from 'twenty-ui/utilities'; import { useScreenSize } from 'twenty-ui/utilities';
const StyledLayout = styled.div` const StyledLayout = styled.div`
@ -63,8 +65,7 @@ export const DefaultLayout = () => {
const windowsWidth = useScreenSize().width; const windowsWidth = useScreenSize().width;
const showAuthModal = useShowAuthModal(); const showAuthModal = useShowAuthModal();
const useShowFullScreen = useShowFullscreen(); const useShowFullScreen = useShowFullscreen();
const [searchParams] = useSearchParams(); const animateModal = useRecoilValue(animateModalState);
const animateModal = searchParams.get('animateModal') !== 'false';
return ( return (
<> <>

View File

@ -1,6 +1,7 @@
import { DEFAULT_WORKSPACE_LOGO } from '@/ui/navigation/navigation-drawer/constants/DefaultWorkspaceLogo'; import { DEFAULT_WORKSPACE_LOGO } from '@/ui/navigation/navigation-drawer/constants/DefaultWorkspaceLogo';
import { useAuth } from '@/auth/hooks/useAuth'; import { useAuth } from '@/auth/hooks/useAuth';
import { animateModalState } from '@/auth/states/animateModalState';
import { currentWorkspaceState } from '@/auth/states/currentWorkspaceState'; import { currentWorkspaceState } from '@/auth/states/currentWorkspaceState';
import { Workspaces, workspacesState } from '@/auth/states/workspaces'; import { Workspaces, workspacesState } from '@/auth/states/workspaces';
import { useBuildWorkspaceUrl } from '@/domain-manager/hooks/useBuildWorkspaceUrl'; import { useBuildWorkspaceUrl } from '@/domain-manager/hooks/useBuildWorkspaceUrl';
@ -66,6 +67,7 @@ export const MultiWorkspaceDropdownDefaultComponents = () => {
const setMultiWorkspaceDropdownState = useSetRecoilState( const setMultiWorkspaceDropdownState = useSetRecoilState(
multiWorkspaceDropdownState, multiWorkspaceDropdownState,
); );
const setAnimateModal = useSetRecoilState(animateModalState);
const handleChange = async (workspace: Workspaces[0]) => { const handleChange = async (workspace: Workspaces[0]) => {
redirectToWorkspaceDomain(getWorkspaceUrl(workspace.workspaceUrls)); redirectToWorkspaceDomain(getWorkspaceUrl(workspace.workspaceUrls));
@ -73,13 +75,13 @@ export const MultiWorkspaceDropdownDefaultComponents = () => {
const createWorkspace = () => { const createWorkspace = () => {
signUpInNewWorkspaceMutation({ signUpInNewWorkspaceMutation({
onCompleted: (data) => { onCompleted: async (data) => {
return redirectToWorkspaceDomain( setAnimateModal(false);
return await redirectToWorkspaceDomain(
getWorkspaceUrl(data.signUpInNewWorkspace.workspace.workspaceUrls), getWorkspaceUrl(data.signUpInNewWorkspace.workspace.workspaceUrls),
AppPath.Verify, AppPath.Verify,
{ {
loginToken: data.signUpInNewWorkspace.loginToken.token, loginToken: data.signUpInNewWorkspace.loginToken.token,
animateModal: false,
}, },
'_blank', '_blank',
); );

View File

@ -1,20 +1,20 @@
import { DropdownMenuItemsContainer } from '@/ui/layout/dropdown/components/DropdownMenuItemsContainer';
import { DEFAULT_WORKSPACE_LOGO } from '@/ui/navigation/navigation-drawer/constants/DefaultWorkspaceLogo';
import { DropdownMenuSeparator } from '@/ui/layout/dropdown/components/DropdownMenuSeparator';
import { getWorkspaceUrl } from '~/utils/getWorkspaceUrl';
import { currentWorkspaceState } from '@/auth/states/currentWorkspaceState'; import { currentWorkspaceState } from '@/auth/states/currentWorkspaceState';
import { Workspaces, workspacesState } from '@/auth/states/workspaces'; import { Workspaces, workspacesState } from '@/auth/states/workspaces';
import { useBuildWorkspaceUrl } from '@/domain-manager/hooks/useBuildWorkspaceUrl'; import { useBuildWorkspaceUrl } from '@/domain-manager/hooks/useBuildWorkspaceUrl';
import { useRecoilValue, useSetRecoilState } from 'recoil';
import { useRedirectToWorkspaceDomain } from '@/domain-manager/hooks/useRedirectToWorkspaceDomain'; import { useRedirectToWorkspaceDomain } from '@/domain-manager/hooks/useRedirectToWorkspaceDomain';
import { useLingui } from '@lingui/react/macro';
import { multiWorkspaceDropdownState } from '@/ui/navigation/navigation-drawer/states/multiWorkspaceDropdownState';
import { useState } from 'react';
import { DropdownMenuSearchInput } from '@/ui/layout/dropdown/components/DropdownMenuSearchInput';
import { DropdownMenuHeader } from '@/ui/layout/dropdown/components/DropdownMenuHeader/DropdownMenuHeader'; import { DropdownMenuHeader } from '@/ui/layout/dropdown/components/DropdownMenuHeader/DropdownMenuHeader';
import { DropdownMenuHeaderLeftComponent } from '@/ui/layout/dropdown/components/DropdownMenuHeader/internal/DropdownMenuHeaderLeftComponent'; import { DropdownMenuHeaderLeftComponent } from '@/ui/layout/dropdown/components/DropdownMenuHeader/internal/DropdownMenuHeaderLeftComponent';
import { DropdownMenuItemsContainer } from '@/ui/layout/dropdown/components/DropdownMenuItemsContainer';
import { DropdownMenuSearchInput } from '@/ui/layout/dropdown/components/DropdownMenuSearchInput';
import { DropdownMenuSeparator } from '@/ui/layout/dropdown/components/DropdownMenuSeparator';
import { DEFAULT_WORKSPACE_LOGO } from '@/ui/navigation/navigation-drawer/constants/DefaultWorkspaceLogo';
import { multiWorkspaceDropdownState } from '@/ui/navigation/navigation-drawer/states/multiWorkspaceDropdownState';
import { useLingui } from '@lingui/react/macro';
import { useState } from 'react';
import { useRecoilValue, useSetRecoilState } from 'recoil';
import { Avatar, IconChevronLeft } from 'twenty-ui/display'; import { Avatar, IconChevronLeft } from 'twenty-ui/display';
import { MenuItemSelectAvatar, UndecoratedLink } from 'twenty-ui/navigation'; import { MenuItemSelectAvatar, UndecoratedLink } from 'twenty-ui/navigation';
import { getWorkspaceUrl } from '~/utils/getWorkspaceUrl';
export const MultiWorkspaceDropdownWorkspacesListComponents = () => { export const MultiWorkspaceDropdownWorkspacesListComponents = () => {
const currentWorkspace = useRecoilValue(currentWorkspaceState); const currentWorkspace = useRecoilValue(currentWorkspaceState);
@ -24,7 +24,7 @@ export const MultiWorkspaceDropdownWorkspacesListComponents = () => {
const { t } = useLingui(); const { t } = useLingui();
const handleChange = async (workspace: Workspaces[0]) => { const handleChange = async (workspace: Workspaces[0]) => {
redirectToWorkspaceDomain(getWorkspaceUrl(workspace.workspaceUrls)); await redirectToWorkspaceDomain(getWorkspaceUrl(workspace.workspaceUrls));
}; };
const setMultiWorkspaceDropdownState = useSetRecoilState( const setMultiWorkspaceDropdownState = useSetRecoilState(
multiWorkspaceDropdownState, multiWorkspaceDropdownState,

View File

@ -2,7 +2,7 @@ import styled from '@emotion/styled';
import { motion } from 'framer-motion'; import { motion } from 'framer-motion';
const StyledContentContainer = styled(motion.div)` const StyledContentContainer = styled(motion.div)`
height: 300px; height: 480px;
margin-bottom: ${({ theme }) => theme.spacing(8)}; margin-bottom: ${({ theme }) => theme.spacing(8)};
margin-top: ${({ theme }) => theme.spacing(4)}; margin-top: ${({ theme }) => theme.spacing(4)};
`; `;

View File

@ -14,11 +14,14 @@ import { useRecoilState, useRecoilValue } from 'recoil';
import { isDefined } from 'twenty-shared/utils'; import { isDefined } from 'twenty-shared/utils';
import { Loader } from 'twenty-ui/feedback'; import { Loader } from 'twenty-ui/feedback';
import { CardPicker, MainButton } from 'twenty-ui/input'; import { CardPicker, MainButton } from 'twenty-ui/input';
import { CAL_LINK, ClickToActionLink } from 'twenty-ui/navigation'; import {
CAL_LINK,
ClickToActionLink,
TWENTY_PRICING_LINK,
} from 'twenty-ui/navigation';
import { import {
BillingPlanKey, BillingPlanKey,
BillingPriceLicensedDto, BillingPriceLicensedDto,
SubscriptionInterval,
useBillingBaseProductPricesQuery, useBillingBaseProductPricesQuery,
} from '~/generated/graphql'; } from '~/generated/graphql';
@ -94,7 +97,7 @@ export const ChooseYourPlan = () => {
const { data: plans } = useBillingBaseProductPricesQuery(); const { data: plans } = useBillingBaseProductPricesQuery();
const currentPlan = billingCheckoutSession.plan || BillingPlanKey.PRO; const currentPlan = billingCheckoutSession.plan;
const getPlanBenefits = (planKey: BillingPlanKey) => { const getPlanBenefits = (planKey: BillingPlanKey) => {
if (planKey === BillingPlanKey.ENTERPRISE) { if (planKey === BillingPlanKey.ENTERPRISE) {
@ -128,7 +131,7 @@ export const ChooseYourPlan = () => {
const baseProductPrice = baseProduct?.prices?.find( const baseProductPrice = baseProduct?.prices?.find(
(price): price is BillingPriceLicensedDto => (price): price is BillingPriceLicensedDto =>
isBillingPriceLicensed(price) && isBillingPriceLicensed(price) &&
price.recurringInterval === SubscriptionInterval.Month, price.recurringInterval === billingCheckoutSession.interval,
); );
const hasWithoutCreditCardTrialPeriod = billing?.trialPeriods.some( const hasWithoutCreditCardTrialPeriod = billing?.trialPeriods.some(
@ -160,27 +163,10 @@ export const ChooseYourPlan = () => {
}; };
}; };
const handleSwitchPlan = (planKey: BillingPlanKey) => {
return () => {
if (isDefined(baseProductPrice)) {
setBillingCheckoutSession({
plan: planKey,
interval: baseProductPrice.recurringInterval,
requirePaymentMethod: billingCheckoutSession.requirePaymentMethod,
});
}
};
};
const { signOut } = useAuth(); const { signOut } = useAuth();
const withCreditCardTrialPeriodDuration = withCreditCardTrialPeriod?.duration; const withCreditCardTrialPeriodDuration = withCreditCardTrialPeriod?.duration;
const alternatePlan =
currentPlan === BillingPlanKey.PRO
? BillingPlanKey.ENTERPRISE
: BillingPlanKey.PRO;
const planName = plans?.plans.find((plan) => plan.planKey === currentPlan) const planName = plans?.plans.find((plan) => plan.planKey === currentPlan)
?.baseProduct.name; ?.baseProduct.name;
@ -252,8 +238,8 @@ export const ChooseYourPlan = () => {
<Trans>Log out</Trans> <Trans>Log out</Trans>
</ClickToActionLink> </ClickToActionLink>
<span /> <span />
<ClickToActionLink onClick={handleSwitchPlan(alternatePlan)}> <ClickToActionLink href={TWENTY_PRICING_LINK}>
<Trans>Switch Plan</Trans> <Trans>Change Plan</Trans>
</ClickToActionLink> </ClickToActionLink>
<span /> <span />
<ClickToActionLink href={CAL_LINK} target="_blank" rel="noreferrer"> <ClickToActionLink href={CAL_LINK} target="_blank" rel="noreferrer">

View File

@ -1,30 +1,30 @@
import { ApolloError } from '@apollo/client';
import { import {
CurrentWorkspace, CurrentWorkspace,
currentWorkspaceState, currentWorkspaceState,
} from '@/auth/states/currentWorkspaceState'; } from '@/auth/states/currentWorkspaceState';
import { useRecoilState } from 'recoil'; import { useRedirectToWorkspaceDomain } from '@/domain-manager/hooks/useRedirectToWorkspaceDomain';
import { SaveAndCancelButtons } from '@/settings/components/SaveAndCancelButtons/SaveAndCancelButtons'; import { SaveAndCancelButtons } from '@/settings/components/SaveAndCancelButtons/SaveAndCancelButtons';
import { SettingsPageContainer } from '@/settings/components/SettingsPageContainer';
import { SettingsPath } from '@/types/SettingsPath';
import { SnackBarVariant } from '@/ui/feedback/snack-bar-manager/components/SnackBar'; import { SnackBarVariant } from '@/ui/feedback/snack-bar-manager/components/SnackBar';
import { useSnackBar } from '@/ui/feedback/snack-bar-manager/hooks/useSnackBar'; import { useSnackBar } from '@/ui/feedback/snack-bar-manager/hooks/useSnackBar';
import { SubMenuTopBarContainer } from '@/ui/layout/page/components/SubMenuTopBarContainer';
import { useIsFeatureEnabled } from '@/workspace/hooks/useIsFeatureEnabled';
import { ApolloError } from '@apollo/client';
import { zodResolver } from '@hookform/resolvers/zod';
import { Trans, useLingui } from '@lingui/react/macro';
import { FormProvider, useForm } from 'react-hook-form';
import { useRecoilState } from 'recoil';
import { isDefined } from 'twenty-shared/utils';
import { z } from 'zod';
import { import {
FeatureFlagKey, FeatureFlagKey,
useUpdateWorkspaceMutation, useUpdateWorkspaceMutation,
} from '~/generated/graphql'; } from '~/generated/graphql';
import { useRedirectToWorkspaceDomain } from '@/domain-manager/hooks/useRedirectToWorkspaceDomain'; import { useNavigateSettings } from '~/hooks/useNavigateSettings';
import { SettingsCustomDomain } from '~/pages/settings/workspace/SettingsCustomDomain'; import { SettingsCustomDomain } from '~/pages/settings/workspace/SettingsCustomDomain';
import { SettingsSubdomain } from '~/pages/settings/workspace/SettingsSubdomain'; import { SettingsSubdomain } from '~/pages/settings/workspace/SettingsSubdomain';
import { useIsFeatureEnabled } from '@/workspace/hooks/useIsFeatureEnabled';
import { useNavigateSettings } from '~/hooks/useNavigateSettings';
import { getSettingsPath } from '~/utils/navigation/getSettingsPath'; import { getSettingsPath } from '~/utils/navigation/getSettingsPath';
import { Trans, useLingui } from '@lingui/react/macro';
import { z } from 'zod';
import { FormProvider, useForm } from 'react-hook-form';
import { zodResolver } from '@hookform/resolvers/zod';
import { SubMenuTopBarContainer } from '@/ui/layout/page/components/SubMenuTopBarContainer';
import { SettingsPath } from '@/types/SettingsPath';
import { SettingsPageContainer } from '@/settings/components/SettingsPageContainer';
import { isDefined } from 'twenty-shared/utils';
export const SettingsDomain = () => { export const SettingsDomain = () => {
const navigate = useNavigateSettings(); const navigate = useNavigateSettings();
@ -151,7 +151,7 @@ export const SettingsDomain = () => {
variant: SnackBarVariant.Error, variant: SnackBarVariant.Error,
}); });
}, },
onCompleted: () => { onCompleted: async () => {
const currentUrl = new URL(window.location.href); const currentUrl = new URL(window.location.href);
currentUrl.hostname = new URL( currentUrl.hostname = new URL(
@ -167,7 +167,7 @@ export const SettingsDomain = () => {
variant: SnackBarVariant.Success, variant: SnackBarVariant.Success,
}); });
redirectToWorkspaceDomain(currentUrl.toString()); await redirectToWorkspaceDomain(currentUrl.toString());
}, },
}); });
}; };

View File

@ -1,8 +1,8 @@
import omit from 'lodash.omit'; import omit from 'lodash.omit';
import { AtomEffect } from 'recoil'; import { AtomEffect } from 'recoil';
import { isDefined } from 'twenty-shared/utils';
import { z } from 'zod'; import { z } from 'zod';
import { cookieStorage } from '~/utils/cookie-storage'; import { cookieStorage } from '~/utils/cookie-storage';
import { isDefined } from 'twenty-shared/utils';
export const localStorageEffect = export const localStorageEffect =
<T>(key?: string): AtomEffect<T> => <T>(key?: string): AtomEffect<T> =>

View File

@ -17,6 +17,7 @@ export { LinkType, SocialLink } from './link/components/SocialLink';
export { UndecoratedLink } from './link/components/UndecoratedLink'; export { UndecoratedLink } from './link/components/UndecoratedLink';
export { CAL_LINK } from './link/constants/Cal'; export { CAL_LINK } from './link/constants/Cal';
export { GITHUB_LINK } from './link/constants/GithubLink'; export { GITHUB_LINK } from './link/constants/GithubLink';
export { TWENTY_PRICING_LINK } from './link/constants/TwentyPricingLink';
export type { export type {
MenuItemIconButton, MenuItemIconButton,
MenuItemProps, MenuItemProps,

View File

@ -0,0 +1 @@
export const TWENTY_PRICING_LINK = 'https://twenty.com/pricing';