0
Fork 0
mirror of https://github.com/logto-io/logto.git synced 2025-01-20 21:32:31 -05:00

refactor(console,phrases): optimize onboarding process (#4214)

This commit is contained in:
Darcy Ye 2023-07-25 01:10:13 +08:00 committed by GitHub
parent 36b5958935
commit ae4f85bf10
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
25 changed files with 163 additions and 481 deletions

View file

@ -9,7 +9,7 @@ type Props = {
children: ReactNode; children: ReactNode;
}; };
const totalSteps = 3; const totalSteps = 2;
function ActionBar({ step, children }: Props) { function ActionBar({ step, children }: Props) {
return ( return (

View file

@ -19,7 +19,6 @@ import { gtagAwTrackingId, gtagSignUpConversionId, logtoProductionHostname } fro
import AppContent from './containers/AppContent'; import AppContent from './containers/AppContent';
import useUserOnboardingData from './hooks/use-user-onboarding-data'; import useUserOnboardingData from './hooks/use-user-onboarding-data';
import * as styles from './index.module.scss'; import * as styles from './index.module.scss';
import About from './pages/About';
import Congrats from './pages/Congrats'; import Congrats from './pages/Congrats';
import SignInExperience from './pages/SignInExperience'; import SignInExperience from './pages/SignInExperience';
import Welcome from './pages/Welcome'; import Welcome from './pages/Welcome';
@ -110,7 +109,6 @@ export function OnboardingRoutes() {
<Route path={OnboardingRoute.Onboarding} element={<AppContent />}> <Route path={OnboardingRoute.Onboarding} element={<AppContent />}>
<Route index element={<Navigate replace to={OnboardingPage.Welcome} />} /> <Route index element={<Navigate replace to={OnboardingPage.Welcome} />} />
<Route path={OnboardingPage.Welcome} element={<Welcome />} /> <Route path={OnboardingPage.Welcome} element={<Welcome />} />
<Route path={OnboardingPage.AboutUser} element={<About />} />
<Route path={OnboardingPage.SignInExperience} element={<SignInExperience />} /> <Route path={OnboardingPage.SignInExperience} element={<SignInExperience />} />
<Route path={OnboardingPage.Congrats} element={<Congrats />} /> <Route path={OnboardingPage.Congrats} element={<Congrats />} />
</Route> </Route>

View file

@ -1,31 +0,0 @@
@use '@/scss/underscore' as _;
.title {
font: var(--font-title-1);
margin-top: _.unit(6);
}
.description {
font: var(--font-body-2);
margin-top: _.unit(3);
}
.form {
width: 100%;
margin-top: _.unit(6);
.titleSelector {
grid-template-columns: repeat(6, 1fr);
align-items: center;
}
.option {
display: flex;
justify-content: center;
min-height: 60px;
}
.cardFieldHeadline {
margin-bottom: _.unit(2);
}
}

View file

@ -1,150 +0,0 @@
import { withAppInsights } from '@logto/app-insights/react';
import { conditional } from '@silverhand/essentials';
import { useEffect } from 'react';
import { Controller, useForm } from 'react-hook-form';
import { useTranslation } from 'react-i18next';
import Case from '@/assets/icons/case.svg';
import PageMeta from '@/components/PageMeta';
import Button from '@/ds-components/Button';
import FormField from '@/ds-components/FormField';
import OverlayScrollbar from '@/ds-components/OverlayScrollbar';
import TextInput from '@/ds-components/TextInput';
import useTenantPathname from '@/hooks/use-tenant-pathname';
import useUserOnboardingData from '@/onboarding/hooks/use-user-onboarding-data';
import * as pageLayout from '@/onboarding/scss/layout.module.scss';
import { trySubmitSafe } from '@/utils/form';
import ActionBar from '../../components/ActionBar';
import { CardSelector, MultiCardSelector } from '../../components/CardSelector';
import type { Questionnaire } from '../../types';
import { OnboardingPage } from '../../types';
import { getOnboardingPage } from '../../utils';
import * as styles from './index.module.scss';
import { titleOptions, companySizeOptions, reasonOptions } from './options';
function About() {
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
const { navigate } = useTenantPathname();
const {
data: { questionnaire },
isBusinessPlan,
update,
} = useUserOnboardingData();
const { control, register, handleSubmit, reset } = useForm<Questionnaire>({
mode: 'onChange',
});
useEffect(() => {
reset(questionnaire);
}, [questionnaire, reset]);
const onSubmit = handleSubmit(
trySubmitSafe(async (formData) => {
await update({ questionnaire: formData });
})
);
const onNext = async () => {
await onSubmit();
navigate(getOnboardingPage(OnboardingPage.SignInExperience));
};
const onBack = async () => {
navigate(getOnboardingPage(OnboardingPage.Welcome));
};
return (
<div className={pageLayout.page}>
<PageMeta titleKey={['cloud.about.page_title', 'cloud.general.onboarding']} />
<OverlayScrollbar className={pageLayout.contentContainer}>
<div className={pageLayout.content}>
<Case />
<div className={styles.title}>{t('cloud.about.title')}</div>
<div className={styles.description}>{t('cloud.about.description')}</div>
<form className={styles.form}>
{isBusinessPlan && (
<>
<FormField
isMultiple
title="cloud.about.title_field"
headlineClassName={styles.cardFieldHeadline}
>
<Controller
control={control}
name="titles"
render={({ field: { onChange, value } }) => (
<MultiCardSelector
className={styles.titleSelector}
optionClassName={styles.option}
value={value ?? []}
options={titleOptions}
onChange={(value) => {
onChange(value.length === 0 ? undefined : value);
}}
/>
)}
/>
</FormField>
<FormField title="cloud.about.company_name_field">
<TextInput
placeholder={t('cloud.about.company_name_placeholder')}
{...register('companyName')}
/>
</FormField>
<FormField
title="cloud.about.company_size_field"
headlineClassName={styles.cardFieldHeadline}
>
<Controller
control={control}
name="companySize"
render={({ field: { onChange, value, name } }) => (
<CardSelector
name={name}
value={value ?? ''}
options={companySizeOptions}
optionClassName={styles.option}
onChange={(value) => {
onChange(conditional(value && value));
}}
/>
)}
/>
</FormField>
</>
)}
<FormField
isMultiple
title="cloud.about.reason_field"
headlineClassName={styles.cardFieldHeadline}
>
<Controller
control={control}
name="reasons"
render={({ field: { onChange, value } }) => (
<MultiCardSelector
value={value ?? []}
options={reasonOptions}
onChange={(value) => {
onChange(value.length === 0 ? undefined : value);
}}
/>
)}
/>
</FormField>
</form>
</div>
</OverlayScrollbar>
<ActionBar step={2}>
<Button title="general.next" type="primary" onClick={onNext} />
<Button title="general.back" onClick={onBack} />
</ActionBar>
</div>
);
}
export default withAppInsights(About);

View file

@ -1,32 +0,0 @@
import type {
CardSelectorOption,
MultiCardSelectorOption,
} from '@/onboarding/components/CardSelector';
import { CompanySize, Reason, Title } from '../../types';
export const titleOptions: MultiCardSelectorOption[] = [
{ title: 'cloud.about.title_options.developer', value: Title.Developer },
{ title: 'cloud.about.title_options.team_lead', value: Title.TeamLead },
{ title: 'cloud.about.title_options.ceo', value: Title.Ceo },
{ title: 'cloud.about.title_options.cto', value: Title.Cto },
{ title: 'cloud.about.title_options.product', value: Title.Product },
{ title: 'cloud.about.title_options.others', value: Title.Others },
];
export const companySizeOptions: CardSelectorOption[] = [
{ title: 'cloud.about.company_options.size_1', value: CompanySize.Scale1 },
{ title: 'cloud.about.company_options.size_2_49', value: CompanySize.Scale2 },
{ title: 'cloud.about.company_options.size_50_199', value: CompanySize.Scale3 },
{ title: 'cloud.about.company_options.size_200_999', value: CompanySize.Scale4 },
{ title: 'cloud.about.company_options.size_1000_plus', value: CompanySize.Scale5 },
];
export const reasonOptions: MultiCardSelectorOption[] = [
{ title: 'cloud.about.reason_options.passwordless', value: Reason.Passwordless },
{ title: 'cloud.about.reason_options.efficiency', value: Reason.Efficiency },
{ title: 'cloud.about.reason_options.access_control', value: Reason.AccessControl },
{ title: 'cloud.about.reason_options.multi_tenancy', value: Reason.MultiTenancy },
{ title: 'cloud.about.reason_options.enterprise', value: Reason.Enterprise },
{ title: 'cloud.about.reason_options.others', value: Reason.Others },
];

View file

@ -218,7 +218,7 @@ function SignInExperience() {
/> />
</div> </div>
</OverlayScrollbar> </OverlayScrollbar>
<ActionBar step={3}> <ActionBar step={2}>
<div className={styles.continueActions}> <div className={styles.continueActions}>
<Button <Button
type="outline" type="outline"
@ -249,7 +249,7 @@ function SignInExperience() {
title="general.back" title="general.back"
disabled={isSubmitting} disabled={isSubmitting}
onClick={() => { onClick={() => {
navigate(getOnboardingPage(OnboardingPage.AboutUser)); navigate(getOnboardingPage(OnboardingPage.Welcome));
}} }}
/> />
</ActionBar> </ActionBar>

View file

@ -1,30 +1,30 @@
@use '@/scss/underscore' as _; @use '@/scss/underscore' as _;
.content {
align-items: center;
}
.congrats {
width: 160px;
height: 160px;
}
.title { .title {
margin-top: _.unit(6);
font: var(--font-title-1); font: var(--font-title-1);
text-align: center; margin-top: _.unit(6);
} }
.description { .description {
margin-top: _.unit(3);
text-align: center;
font: var(--font-body-2); font: var(--font-body-2);
margin-top: _.unit(3);
} }
.form { .form {
width: 100%; width: 100%;
margin-top: _.unit(6); margin-top: _.unit(6);
.titleSelector {
grid-template-columns: repeat(6, 1fr);
align-items: center;
}
.option {
display: flex;
justify-content: center;
min-height: 60px;
}
.cardFieldHeadline { .cardFieldHeadline {
margin-bottom: _.unit(2); margin-bottom: _.unit(2);
} }

View file

@ -1,27 +1,28 @@
import { withAppInsights } from '@logto/app-insights/react'; import { withAppInsights } from '@logto/app-insights/react';
import classNames from 'classnames'; import { conditional } from '@silverhand/essentials';
import { useEffect } from 'react'; import { useEffect } from 'react';
import { Controller, useForm } from 'react-hook-form'; import { Controller, useForm } from 'react-hook-form';
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
import WelcomeImage from '@/assets/images/sign-in-experience-welcome.svg'; import Case from '@/assets/icons/case.svg';
import PageMeta from '@/components/PageMeta'; import PageMeta from '@/components/PageMeta';
import Button from '@/ds-components/Button'; import Button from '@/ds-components/Button';
import FormField from '@/ds-components/FormField'; import FormField from '@/ds-components/FormField';
import OverlayScrollbar from '@/ds-components/OverlayScrollbar'; import OverlayScrollbar from '@/ds-components/OverlayScrollbar';
import TextInput from '@/ds-components/TextInput';
import useTenantPathname from '@/hooks/use-tenant-pathname'; import useTenantPathname from '@/hooks/use-tenant-pathname';
import ActionBar from '@/onboarding/components/ActionBar';
import { CardSelector } from '@/onboarding/components/CardSelector';
import useUserOnboardingData from '@/onboarding/hooks/use-user-onboarding-data'; import useUserOnboardingData from '@/onboarding/hooks/use-user-onboarding-data';
import * as pageLayout from '@/onboarding/scss/layout.module.scss'; import * as pageLayout from '@/onboarding/scss/layout.module.scss';
import { trySubmitSafe } from '@/utils/form'; import { trySubmitSafe } from '@/utils/form';
import ActionBar from '../../components/ActionBar';
import { CardSelector, MultiCardSelector } from '../../components/CardSelector';
import type { Questionnaire } from '../../types'; import type { Questionnaire } from '../../types';
import { OnboardingPage } from '../../types'; import { OnboardingPage, Project } from '../../types';
import { getOnboardingPage } from '../../utils'; import { getOnboardingPage } from '../../utils';
import * as styles from './index.module.scss'; import * as styles from './index.module.scss';
import { deploymentTypeOptions, projectOptions } from './options'; import { titleOptions, companySizeOptions, reasonOptions, projectOptions } from './options';
function Welcome() { function Welcome() {
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' }); const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
@ -32,12 +33,9 @@ function Welcome() {
update, update,
} = useUserOnboardingData(); } = useUserOnboardingData();
const { const { control, register, handleSubmit, reset, watch } = useForm<Questionnaire>({
control, mode: 'onChange',
handleSubmit, });
formState: { isSubmitting, isValid },
reset,
} = useForm<Questionnaire>({ defaultValues: questionnaire, mode: 'onChange' });
useEffect(() => { useEffect(() => {
reset(questionnaire); reset(questionnaire);
@ -51,15 +49,15 @@ function Welcome() {
const onNext = async () => { const onNext = async () => {
await onSubmit(); await onSubmit();
navigate(getOnboardingPage(OnboardingPage.AboutUser)); navigate(getOnboardingPage(OnboardingPage.SignInExperience));
}; };
return ( return (
<div className={pageLayout.page}> <div className={pageLayout.page}>
<PageMeta titleKey={['cloud.welcome.page_title', 'cloud.general.onboarding']} /> <PageMeta titleKey={['cloud.welcome.page_title', 'cloud.general.onboarding']} />
<OverlayScrollbar className={pageLayout.contentContainer}> <OverlayScrollbar className={pageLayout.contentContainer}>
<div className={classNames(pageLayout.content, styles.content)}> <div className={pageLayout.content}>
<WelcomeImage className={styles.congrats} /> <Case />
<div className={styles.title}>{t('cloud.welcome.title')}</div> <div className={styles.title}>{t('cloud.welcome.title')}</div>
<div className={styles.description}>{t('cloud.welcome.description')}</div> <div className={styles.description}>{t('cloud.welcome.description')}</div>
<form className={styles.form}> <form className={styles.form}>
@ -81,20 +79,73 @@ function Welcome() {
)} )}
/> />
</FormField> </FormField>
{/* Check whether it is a business use case */}
{watch('project') === Project.Company && (
<>
<FormField
isMultiple
title="cloud.welcome.title_field"
headlineClassName={styles.cardFieldHeadline}
>
<Controller
control={control}
name="titles"
render={({ field: { onChange, value } }) => (
<MultiCardSelector
className={styles.titleSelector}
optionClassName={styles.option}
value={value ?? []}
options={titleOptions}
onChange={(value) => {
onChange(value.length === 0 ? undefined : value);
}}
/>
)}
/>
</FormField>
<FormField title="cloud.welcome.company_name_field">
<TextInput
placeholder={t('cloud.welcome.company_name_placeholder')}
{...register('companyName')}
/>
</FormField>
<FormField
title="cloud.welcome.company_size_field"
headlineClassName={styles.cardFieldHeadline}
>
<Controller
control={control}
name="companySize"
render={({ field: { onChange, value, name } }) => (
<CardSelector
name={name}
value={value ?? ''}
options={companySizeOptions}
optionClassName={styles.option}
onChange={(value) => {
onChange(conditional(value && value));
}}
/>
)}
/>
</FormField>
</>
)}
<FormField <FormField
title="cloud.welcome.deployment_type_field" isMultiple
title="cloud.welcome.reason_field"
headlineClassName={styles.cardFieldHeadline} headlineClassName={styles.cardFieldHeadline}
> >
<Controller <Controller
control={control} control={control}
name="deploymentType" name="reasons"
rules={{ required: true }} render={({ field: { onChange, value } }) => (
render={({ field: { onChange, value, name } }) => ( <MultiCardSelector
<CardSelector value={value ?? []}
name={name} options={reasonOptions}
value={value} onChange={(value) => {
options={deploymentTypeOptions} onChange(value.length === 0 ? undefined : value);
onChange={onChange} }}
/> />
)} )}
/> />
@ -103,12 +154,7 @@ function Welcome() {
</div> </div>
</OverlayScrollbar> </OverlayScrollbar>
<ActionBar step={1}> <ActionBar step={1}>
<Button <Button title="general.next" type="primary" onClick={onNext} />
title="general.next"
type="primary"
disabled={isSubmitting || !isValid}
onClick={onNext}
/>
</ActionBar> </ActionBar>
</div> </div>
); );

View file

@ -1,10 +1,11 @@
import Building from '@/assets/icons/building.svg'; import Building from '@/assets/icons/building.svg';
import Cloud from '@/assets/icons/cloud.svg';
import Database from '@/assets/icons/database.svg';
import Pizza from '@/assets/icons/pizza.svg'; import Pizza from '@/assets/icons/pizza.svg';
import type { CardSelectorOption } from '@/onboarding/components/CardSelector'; import type {
CardSelectorOption,
MultiCardSelectorOption,
} from '@/onboarding/components/CardSelector';
import { DeploymentType, Project } from '../../types'; import { Project, CompanySize, Reason, Title } from '../../types';
export const projectOptions: CardSelectorOption[] = [ export const projectOptions: CardSelectorOption[] = [
{ {
@ -19,15 +20,28 @@ export const projectOptions: CardSelectorOption[] = [
}, },
]; ];
export const deploymentTypeOptions: CardSelectorOption[] = [ export const titleOptions: MultiCardSelectorOption[] = [
{ { title: 'cloud.welcome.title_options.developer', value: Title.Developer },
icon: <Database />, { title: 'cloud.welcome.title_options.team_lead', value: Title.TeamLead },
title: 'cloud.welcome.deployment_type_options.open_source', { title: 'cloud.welcome.title_options.ceo', value: Title.Ceo },
value: DeploymentType.OpenSource, { title: 'cloud.welcome.title_options.cto', value: Title.Cto },
}, { title: 'cloud.welcome.title_options.product', value: Title.Product },
{ { title: 'cloud.welcome.title_options.others', value: Title.Others },
icon: <Cloud />, ];
title: 'cloud.welcome.deployment_type_options.cloud',
value: DeploymentType.Cloud, export const companySizeOptions: CardSelectorOption[] = [
}, { title: 'cloud.welcome.company_options.size_1', value: CompanySize.Scale1 },
{ title: 'cloud.welcome.company_options.size_2_49', value: CompanySize.Scale2 },
{ title: 'cloud.welcome.company_options.size_50_199', value: CompanySize.Scale3 },
{ title: 'cloud.welcome.company_options.size_200_999', value: CompanySize.Scale4 },
{ title: 'cloud.welcome.company_options.size_1000_plus', value: CompanySize.Scale5 },
];
export const reasonOptions: MultiCardSelectorOption[] = [
{ title: 'cloud.welcome.reason_options.passwordless', value: Reason.Passwordless },
{ title: 'cloud.welcome.reason_options.efficiency', value: Reason.Efficiency },
{ title: 'cloud.welcome.reason_options.access_control', value: Reason.AccessControl },
{ title: 'cloud.welcome.reason_options.multi_tenancy', value: Reason.MultiTenancy },
{ title: 'cloud.welcome.reason_options.enterprise', value: Reason.Enterprise },
{ title: 'cloud.welcome.reason_options.others', value: Reason.Others },
]; ];

View file

@ -7,6 +7,7 @@ export enum OnboardingRoute {
export enum OnboardingPage { export enum OnboardingPage {
Welcome = 'welcome', Welcome = 'welcome',
/** @deprecated Merged `about-user` to `welcome` page. */
AboutUser = 'about-user', AboutUser = 'about-user',
SignInExperience = 'sign-in-experience', SignInExperience = 'sign-in-experience',
Congrats = 'congrats', Congrats = 'congrats',
@ -17,7 +18,8 @@ export enum Project {
Company = 'company', Company = 'company',
} }
export enum DeploymentType { /** @deprecated Open-source options was for cloud preview use, no longer needed. Use default `Cloud` value for placeholder. */
enum DeploymentType {
OpenSource = 'open-source', OpenSource = 'open-source',
Cloud = 'cloud', Cloud = 'cloud',
} }
@ -50,7 +52,8 @@ export enum Reason {
const questionnaireGuard = z.object({ const questionnaireGuard = z.object({
project: z.nativeEnum(Project), project: z.nativeEnum(Project),
deploymentType: z.nativeEnum(DeploymentType), /** @deprecated Open-source options was for cloud preview use, no longer needed. Use default `Cloud` value for placeholder. */
deploymentType: z.nativeEnum(DeploymentType).optional().default(DeploymentType.Cloud),
titles: z.array(z.nativeEnum(Title)).optional(), titles: z.array(z.nativeEnum(Title)).optional(),
companyName: z.string().optional(), companyName: z.string().optional(),
companySize: z.nativeEnum(CompanySize).optional(), companySize: z.nativeEnum(CompanySize).optional(),

View file

@ -4,25 +4,14 @@ const cloud = {
}, },
welcome: { welcome: {
page_title: 'Willkommen', page_title: 'Willkommen',
title: 'Willkommen, lass uns deine eigene Logto Cloud Preview erstellen', title: 'Willkommen bei Logto Cloud! Wir möchten gerne ein bisschen mehr über Sie erfahren.',
description: description:
'Egal, ob Sie ein Open-Source- oder Cloud-Benutzer sind, machen Sie eine Tour durch die Showcase und erleben Sie den vollen Wert von Logto. Die Cloud-Vorschau dient auch als vorläufige Version von Logto Cloud.', 'Machen Sie Ihre Logto-Erfahrung einzigartig, indem Sie uns besser kennenlernen. Ihre Informationen sind bei uns sicher.',
project_field: 'Ich verwende Logto für', project_field: 'Ich verwende Logto für',
project_options: { project_options: {
personal: 'Persönliches Projekt', personal: 'Persönliches Projekt',
company: 'Unternehmensprojekt', company: 'Unternehmensprojekt',
}, },
deployment_type_field: 'Bevorzugst du Open-Source oder Cloud?',
deployment_type_options: {
open_source: 'Open-Source',
cloud: 'Cloud',
},
},
about: {
page_title: 'Ein bisschen über Sie',
title: 'Ein wenig über Sie',
description:
'Machen Sie Ihre Logto-Erfahrung einzigartig, indem Sie uns besser kennenlernen. Ihre Informationen sind bei uns sicher.',
title_field: 'Ihr Titel', title_field: 'Ihr Titel',
title_options: { title_options: {
developer: 'Entwickler', developer: 'Entwickler',

View file

@ -4,25 +4,14 @@ const cloud = {
}, },
welcome: { welcome: {
page_title: 'Welcome', page_title: 'Welcome',
title: 'Welcome and lets create your own Logto Cloud Preview', title: "Welcome to Logto Cloud! We'd love to learn a bit about you.",
description: description:
'Whether youre an open-source or cloud user, take a tour of the showcase and experience the full value of Logto. Cloud preview also serves as a preliminary version of Logto Cloud.', 'Lets make your Logto experience unique to you by getting to know you better. Your information is safe with us.',
project_field: 'Im using Logto for', project_field: 'Im using Logto for',
project_options: { project_options: {
personal: 'Personal project', personal: 'Personal project',
company: 'Company project', company: 'Company project',
}, },
deployment_type_field: 'Prefer open-source or cloud?',
deployment_type_options: {
open_source: 'Open-Source',
cloud: 'Cloud',
},
},
about: {
page_title: 'A bit about you',
title: 'A little bit about you',
description:
'Lets make your Logto experience unique to you by getting to know you better. Your information is safe with us.',
title_field: 'Your title', title_field: 'Your title',
title_options: { title_options: {
developer: 'Developer', developer: 'Developer',

View file

@ -3,26 +3,15 @@ const cloud = {
onboarding: 'Integración', onboarding: 'Integración',
}, },
welcome: { welcome: {
page_title: 'Bienvenido', page_title: 'Bienvenida',
title: 'Bienvenido y creemos su propia vista previa de Logto Cloud', title: '¡Bienvenido a Logto Cloud! Nos encantaría saber un poco más sobre ti.',
description: description:
'Ya sea que sea un usuario de código abierto o en la nube, realice un recorrido por la vitrina y experimente el valor completo de Logto. La vista previa en la nube también sirve como una versión preliminar de Logto Cloud.', 'Hagamos que su experiencia de Logto sea única para usted al conocerlo mejor. Su información está segura con nosotros.',
project_field: 'Estoy usando Logto para:', project_field: 'Estoy usando Logto para:',
project_options: { project_options: {
personal: 'Proyecto personal', personal: 'Proyecto personal',
company: 'Proyecto empresarial', company: 'Proyecto empresarial',
}, },
deployment_type_field: '¿Prefiere el código abierto o la nube?',
deployment_type_options: {
open_source: 'Código abierto',
cloud: 'Nube',
},
},
about: {
page_title: 'Un poco sobre ti',
title: 'Un poco sobre ti',
description:
'Hagamos que su experiencia de Logto sea única para usted al conocerlo mejor. Su información está segura con nosotros.',
title_field: 'Tu título', title_field: 'Tu título',
title_options: { title_options: {
developer: 'Desarrollador', developer: 'Desarrollador',

View file

@ -3,26 +3,15 @@ const cloud = {
onboarding: 'Intégration', onboarding: 'Intégration',
}, },
welcome: { welcome: {
page_title: 'Bienvenue', page_title: 'Welcome',
title: 'Bienvenue et créons votre propre aperçu cloud de Logto', title: 'Bienvenue dans Logto Cloud ! Nous aimerions en savoir un peu plus sur vous.',
description: description:
'Que vous soyez un utilisateur open-source ou cloud, faites une visite de la vitrine et découvrez la valeur totale de Logto. Le Cloud Preview sert également de version préliminaire de Logto Cloud.', 'Personnalisons votre expérience Logto en vous connaissant mieux. Vos informations sont en sécurité avec nous.',
project_field: "J'utilise Logto pour", project_field: "J'utilise Logto pour",
project_options: { project_options: {
personal: 'Projet personnel', personal: 'Projet personnel',
company: "Projet d'entreprise", company: "Projet d'entreprise",
}, },
deployment_type_field: 'Vous préférez open-source ou cloud?',
deployment_type_options: {
open_source: 'Open-Source',
cloud: 'Cloud',
},
},
about: {
page_title: 'Un peu à propos de vous',
title: 'Un peu à propos de vous',
description:
'Personnalisons votre expérience Logto en vous connaissant mieux. Vos informations sont en sécurité avec nous.',
title_field: 'Votre titre', title_field: 'Votre titre',
title_options: { title_options: {
developer: 'Développeur', developer: 'Développeur',

View file

@ -4,25 +4,14 @@ const cloud = {
}, },
welcome: { welcome: {
page_title: 'Benvenuto', page_title: 'Benvenuto',
title: 'Benvenuto e creiamo insieme la tua anteprima di Logto Cloud', title: "Benvenuto in Logto Cloud! Ci piacerebbe conoscere un po' di te.",
description: description:
'Che tu sia un utente open-source o cloud, fai un tour della vetrina e scopri il valore completo di Logto. Cloud preview serve anche come versione preliminare di Logto Cloud.', 'Facciamo diventare la tua esperienza Logto unica conoscendoti meglio. Le tue informazioni sono al sicuro con noi.',
project_field: 'Sto usando Logto per', project_field: 'Sto usando Logto per',
project_options: { project_options: {
personal: 'Progetto personale', personal: 'Progetto personale',
company: 'Progetto aziendale', company: 'Progetto aziendale',
}, },
deployment_type_field: 'Preferisci open-source o cloud?',
deployment_type_options: {
open_source: 'Open-Source',
cloud: 'Cloud',
},
},
about: {
page_title: "Un po' di te",
title: "Un po' di te per rendere unica la tua esperienza Logto",
description:
'Facciamo diventare la tua esperienza Logto unica conoscendoti meglio. Le tue informazioni sono al sicuro con noi.',
title_field: 'La tua posizione', title_field: 'La tua posizione',
title_options: { title_options: {
developer: 'Sviluppatore', developer: 'Sviluppatore',

View file

@ -4,25 +4,14 @@ const cloud = {
}, },
welcome: { welcome: {
page_title: 'ようこそ', page_title: 'ようこそ',
title: 'ようこそ、あなたのLogto Cloud Previewを作成しましょう', title: 'Logto Cloud へようこそ!あなたについて少し学びたいです。',
description: description:
'オープンソースユーザーでもクラウドユーザーでも、ショーケースを見て、Logtoのすべての価値を体験してください。Cloud PreviewはLogto Cloudの予備バージョンでもあります。', 'あなたの情報を知ることで、あなたにユニークなLogtoエクスペリエンスを提供します。あなたの情報は安全に保管されます。',
project_field: 'ログトを使用しています', project_field: 'ログトを使用しています',
project_options: { project_options: {
personal: '個人プロジェクト', personal: '個人プロジェクト',
company: '会社プロジェクト', company: '会社プロジェクト',
}, },
deployment_type_field: 'オープンソースまたはクラウドを希望しますか?',
deployment_type_options: {
open_source: 'オープンソース',
cloud: 'クラウド',
},
},
about: {
page_title: 'あなたについて',
title: 'あなたについて少し教えてください',
description:
'あなたの情報を知ることで、あなたにユニークなLogtoエクスペリエンスを提供します。あなたの情報は安全に保管されます。',
title_field: 'あなたのタイトル', title_field: 'あなたのタイトル',
title_options: { title_options: {
developer: '開発者', developer: '開発者',

View file

@ -4,25 +4,14 @@ const cloud = {
}, },
welcome: { welcome: {
page_title: '환영합니다', page_title: '환영합니다',
title: '환영합니다. 당신 만의 Logto Cloud Preview를 만들어 보세요', title: 'Logto Cloud 에 오신 것을 환영합니다! 조금 더 알고 싶어요.',
description: description:
'오픈 소스 사용자든 클라우드 사용자든, 쇼케이스를 둘러보고 Logto의 모든 가치를 경험해 보세요. 클라우드 미리 보기는 Logto Cloud의 미리 보기 버전으로도 제공돼요.', '당신에 대해 더 잘 파악하여 Logto 경험을 특별하게 만들어 드릴게요. 정보는 저희가 안전하게 관리해요.',
project_field: 'Logto를 아래의 목적으로 사용해요', project_field: 'Logto 를 아래의 목적으로 사용해요',
project_options: { project_options: {
personal: '개인 프로젝트', personal: '개인 프로젝트',
company: '기업 프로젝트', company: '기업 프로젝트',
}, },
deployment_type_field: '오픈 소스와 클라우드 중 어느 쪽을 선호하시나요?',
deployment_type_options: {
open_source: '오픈 소스',
cloud: '클라우드',
},
},
about: {
page_title: '본인에 대해 간략히 소개하기',
title: '당신에 관하여',
description:
'당신에 대해 더 잘 파악하여 Logto 경험을 특별하게 만들어 드릴게요. 정보는 저희가 안전하게 관리해요.',
title_field: '당신의 직함', title_field: '당신의 직함',
title_options: { title_options: {
developer: '개발자', developer: '개발자',

View file

@ -3,26 +3,15 @@ const cloud = {
onboarding: 'Wdrażanie', onboarding: 'Wdrażanie',
}, },
welcome: { welcome: {
page_title: 'Witaj', page_title: 'Witamy',
title: 'Witaj i stwórz własny podgląd chmury Logto', title: 'Witaj w chmurze Logto! Chcielibyśmy dowiedzieć się trochę o Tobie.',
description: description:
'Nie ważne czy jesteś użytkownikiem open-source czy chmury, przejdź przez prezentacje i doświadcz pełnej wartości Logto. Podgląd chmury serwuje również jako wstępna wersja chmury Logto.', 'Stwórz indywidualne wrażenia z Logto dzięki naszej wiedzy na temat Ciebie. Twoje informacje są bezpieczne u nas.',
project_field: 'Używam Logto do', project_field: 'Używam Logto do',
project_options: { project_options: {
personal: 'Projektu osobistego', personal: 'Projektu osobistego',
company: 'Projektu firmowego', company: 'Projektu firmowego',
}, },
deployment_type_field: 'Wolisz open-source czy chmurę?',
deployment_type_options: {
open_source: 'Open-Source',
cloud: 'Chmura',
},
},
about: {
page_title: 'Trochę o Tobie',
title: 'Trochę o Tobie',
description:
'Stwórz indywidualne wrażenia z Logto dzięki naszej wiedzy na temat Ciebie. Twoje informacje są bezpieczne u nas.',
title_field: 'Twój tytuł', title_field: 'Twój tytuł',
title_options: { title_options: {
developer: 'Developer', developer: 'Developer',

View file

@ -3,26 +3,15 @@ const cloud = {
onboarding: 'Integração', onboarding: 'Integração',
}, },
welcome: { welcome: {
page_title: 'Boas-vindas', page_title: 'Bem-vindo',
title: 'Boas-vindas e vamos criar sua própria visualização de nuvem Logto', title: 'Bem-vindo ao Logto Cloud! Gostaríamos de conhecer um pouco sobre você.',
description: description:
'Se você é um usuário de código aberto ou de nuvem, faça um passeio pela vitrine e experimente o valor completo do Logto. A visualização da nuvem também serve como uma versão preliminar da nuvem Logto.', 'Vamos tornar sua experiência Logto única, conhecendo você melhor. Suas informações estão seguras conosco.',
project_field: 'Estou usando o Logto para', project_field: 'Estou usando o Logto para',
project_options: { project_options: {
personal: 'Projeto pessoal', personal: 'Projeto pessoal',
company: 'Projeto da empresa', company: 'Projeto da empresa',
}, },
deployment_type_field: 'Preferencialmente de código aberto ou na nuvem?',
deployment_type_options: {
open_source: 'Código aberto',
cloud: 'Nuvem',
},
},
about: {
page_title: 'Um pouco sobre você',
title: 'Um pouco sobre você',
description:
'Vamos tornar sua experiência Logto única, conhecendo você melhor. Suas informações estão seguras conosco.',
title_field: 'O seu título', title_field: 'O seu título',
title_options: { title_options: {
developer: 'Desenvolvedor', developer: 'Desenvolvedor',

View file

@ -4,25 +4,14 @@ const cloud = {
}, },
welcome: { welcome: {
page_title: 'Bem-vindo', page_title: 'Bem-vindo',
title: 'Bem-vindo e vamos criar a sua própria visualização da Logto Cloud', title: 'Bem-vindo ao Logto Cloud! Gostaríamos de aprender um pouco sobre você.',
description: description:
'Seja um usuário de código aberto ou de nuvem, faça um passeio pela demonstração e experimente o valor total da Logto. A pré-visualização da nuvem também serve como uma versão preliminar da Logto Cloud.', 'Vamos tornar a experiência da Logto única para você conhecendo você melhor. Suas informações estão seguras conosco.',
project_field: 'Estou usando a Logto para', project_field: 'Estou usando a Logto para',
project_options: { project_options: {
personal: 'Projeto pessoal', personal: 'Projeto pessoal',
company: 'Projeto da empresa', company: 'Projeto da empresa',
}, },
deployment_type_field: 'Prefere código aberto ou nuvem?',
deployment_type_options: {
open_source: 'Código aberto',
cloud: 'Nuvem',
},
},
about: {
page_title: 'Um pouco sobre você',
title: 'Um pouco sobre você',
description:
'Vamos tornar a experiência da Logto única para você conhecendo você melhor. Suas informações estão seguras conosco.',
title_field: 'O seu título', title_field: 'O seu título',
title_options: { title_options: {
developer: 'Desenvolvedor/a', developer: 'Desenvolvedor/a',
@ -119,7 +108,7 @@ const cloud = {
'Você entrou com sucesso usando sua conta social. Para garantir uma integração perfeita e acesso a todos os recursos do Logto, recomendamos que você prossiga para configurar seu próprio conector social.', 'Você entrou com sucesso usando sua conta social. Para garantir uma integração perfeita e acesso a todos os recursos do Logto, recomendamos que você prossiga para configurar seu próprio conector social.',
}, },
tenant: { tenant: {
create_tenant: 'Create tenant', create_tenant: 'Criar novo inquilino',
}, },
}; };

View file

@ -4,26 +4,14 @@ const cloud = {
}, },
welcome: { welcome: {
page_title: 'Добро пожаловать', page_title: 'Добро пожаловать',
title: title: 'Добро пожаловать в облако Logto! Мы хотели бы узнать вас получше.',
'Добро пожаловать, и давайте создадим свой собственный предварительный просмотр облака Logto',
description: description:
'Независимо от того, являетесь ли вы пользователем открытого исходного кода или облачным пользователем, ознакомьтесь с витриной и ощутите полную ценность Logto. Предварительный просмотр облака также служит предварительной версией облака Logto.', 'Сделайте свой опыт работы с Logto уникальным для вас, узнав вас получше. Ваша информация надежно защищена.',
project_field: 'Я использую Logto для', project_field: 'Я использую Logto для',
project_options: { project_options: {
personal: 'Личного проекта', personal: 'Личного проекта',
company: 'Компании', company: 'Компании',
}, },
deployment_type_field: 'Предпочитаете открытый исходный код или облако?',
deployment_type_options: {
open_source: 'Открытый Исходный Код',
cloud: 'Облачный',
},
},
about: {
page_title: 'Немного о вас',
title: 'Немного о вас',
description:
'Сделайте свой опыт работы с Logto уникальным для вас, узнав вас получше. Ваша информация надежно защищена.',
title_field: 'Твоя должность', title_field: 'Твоя должность',
title_options: { title_options: {
developer: 'Разработчик', developer: 'Разработчик',

View file

@ -3,26 +3,15 @@ const cloud = {
onboarding: 'Başlatma', onboarding: 'Başlatma',
}, },
welcome: { welcome: {
page_title: 'Hoşgeldiniz', page_title: 'Hoş geldiniz',
title: 'Hoşgeldiniz ve kendi Logto Cloud Önizlemenizi oluşturalım', title: "Logto Cloud'a hoş geldiniz! Sizi biraz tanımak istiyoruz.",
description: description:
"Açık kaynak veya bulut kullanıcısı olsanız da, sergi turuna çıkın ve Logto'nun tam değerini deneyimleyin. Cloud önizlemesi ayrıca Logto Cloud'un ön sürümü olarak da hizmet verir.", 'Sizi daha iyi tanıyarak Logto deneyiminizi size özel hale getirelim. Bilgileriniz bizimle güvende.',
project_field: "Logto'yu kullanıyorum çünkü", project_field: "Logto'yu kullanıyorum çünkü",
project_options: { project_options: {
personal: 'Kişisel proje', personal: 'Kişisel proje',
company: 'Şirket projesi', company: 'Şirket projesi',
}, },
deployment_type_field: 'Açık kaynak veya bulut tercih eder misiniz?',
deployment_type_options: {
open_source: 'Açık Kaynak',
cloud: 'Bulut',
},
},
about: {
page_title: 'Biraz sen hakkında',
title: 'Senin hakkında biraz bilgi',
description:
'Sizi daha iyi tanıyarak Logto deneyiminizi size özel hale getirelim. Bilgileriniz bizimle güvende.',
title_field: 'Unvanın', title_field: 'Unvanın',
title_options: { title_options: {
developer: 'Geliştirici', developer: 'Geliştirici',

View file

@ -4,24 +4,13 @@ const cloud = {
}, },
welcome: { welcome: {
page_title: '欢迎', page_title: '欢迎',
title: '欢迎来到 Logto Cloud预览版让我们一起创建独属于你的体验', title: '欢迎来到 Logto Cloud我们很想了解你。',
description: description: '通过更好地了解你,我们可以使你的 Logto 体验更加个性化。你的信息是安全的。',
'无论你是开源用户还是云用户,都可以在展示中了解 Logto 的全部价值。Cloud 预览版也是 Logto Cloud 的初步版本。',
project_field: '我使用 Logto 是为了', project_field: '我使用 Logto 是为了',
project_options: { project_options: {
personal: '个人项目', personal: '个人项目',
company: '公司项目', company: '公司项目',
}, },
deployment_type_field: '你偏爱开源还是云?',
deployment_type_options: {
open_source: '开源',
cloud: '云',
},
},
about: {
page_title: '关于你',
title: '关于你的一些信息',
description: '通过更好地了解你,我们可以使你的 Logto 体验更加个性化。你的信息是安全的。',
title_field: '你的头衔', title_field: '你的头衔',
title_options: { title_options: {
developer: '开发人员', developer: '开发人员',

View file

@ -4,24 +4,13 @@ const cloud = {
}, },
welcome: { welcome: {
page_title: '歡迎', page_title: '歡迎',
title: '歡迎來到 Logto Cloud預覽版讓我們一起創建獨屬於你的體驗', title: '歡迎來到 Logto Cloud我們很想了解你的一些信息。',
description: description: '通過更好地了解你,我們可以使你的 Logto 體驗更加個性化。你的信息是安全的。',
'無論你是開源用戶還是雲用戶,都可以在展示中了解 Logto 的全部價值。Cloud 預覽版也是 Logto Cloud 的初步版本。',
project_field: '我使用 Logto 是為了', project_field: '我使用 Logto 是為了',
project_options: { project_options: {
personal: '個人專案', personal: '個人專案',
company: '公司專案', company: '公司專案',
}, },
deployment_type_field: '你偏愛開源還是雲?',
deployment_type_options: {
open_source: '開源',
cloud: '雲',
},
},
about: {
page_title: '關於你',
title: '關於你的一些信息',
description: '通過更好地了解你,我們可以使你的 Logto 體驗更加個性化。你的信息是安全的。',
title_field: '你的頭銜', title_field: '你的頭銜',
title_options: { title_options: {
developer: '開發人員', developer: '開發人員',

View file

@ -4,24 +4,13 @@ const cloud = {
}, },
welcome: { welcome: {
page_title: '歡迎', page_title: '歡迎',
title: '歡迎來到 Logto Cloud預覽版讓我們一起創建獨屬於你的體驗', title: '歡迎來到 Logto Cloud我們很想了解一些關於你的資訊。',
description: description: '通過更好地了解你,我們可以使你的 Logto 體驗更加個性化。你的信息是安全的。',
'無論你是開源用戶還是雲用戶,都可以在展示中了解 Logto 的全部價值。Cloud 預覽版也是 Logto Cloud 的初步版本。',
project_field: '我使用 Logto 是為了', project_field: '我使用 Logto 是為了',
project_options: { project_options: {
personal: '個人專案', personal: '個人專案',
company: '公司專案', company: '公司專案',
}, },
deployment_type_field: '你偏愛開源還是雲?',
deployment_type_options: {
open_source: '開源',
cloud: '雲',
},
},
about: {
page_title: '關於你',
title: '關於你的一些信息',
description: '通過更好地了解你,我們可以使你的 Logto 體驗更加個性化。你的信息是安全的。',
title_field: '你的頭銜', title_field: '你的頭銜',
title_options: { title_options: {
developer: '開發人員', developer: '開發人員',