0
Fork 0
mirror of https://github.com/logto-io/logto.git synced 2024-12-30 20:33:54 -05:00

feat(console): inspire me (#3360)

This commit is contained in:
Xiao Yijun 2023-03-13 13:58:58 +08:00 committed by GitHub
parent fa85b7d0eb
commit b1b5200876
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
33 changed files with 465 additions and 141 deletions

View file

@ -58,6 +58,7 @@ jobs:
env:
INTEGRATION_TEST: true
IS_CLOUD: ${{ contains(matrix.target, 'cloud') && '1' || '0' }}
PATH_BASED_MULTI_TENANCY: ${{ contains(matrix.target, 'cloud') && '1' || '0' }}
steps:
- uses: actions/checkout@v3

View file

@ -0,0 +1,15 @@
<svg width="24" height="24" viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg">
<path d="M16.0723 5.45703C15.3365 4.8519 14.475 4.41865 13.5504 4.18891C12.6259 3.95916 11.6617 3.9387 10.7283 4.12903C9.4854 4.38009 8.34509 4.99469 7.45204 5.89486C6.559 6.79502 5.95347 7.94018 5.7123 9.18503C5.53617 10.1184 5.56818 11.0792 5.80602 11.9987C6.04387 12.9183 6.4817 13.7741 7.0883 14.505C7.65124 15.1403 7.97392 15.9526 8.0003 16.801V19.201C8.0003 19.8376 8.25315 20.448 8.70324 20.8981C9.15333 21.3482 9.76378 21.601 10.4003 21.601H13.6003C14.2368 21.601 14.8473 21.3482 15.2974 20.8981C15.7474 20.448 16.0003 19.8376 16.0003 19.201V16.953C16.0271 16.0164 16.3713 15.1166 16.9763 14.401C18.0364 13.0897 18.5361 11.4129 18.3668 9.73513C18.1976 8.05738 17.373 6.51422 16.0723 5.44103V5.45703ZM14.4003 19.201C14.4003 19.4132 14.316 19.6167 14.166 19.7667C14.016 19.9167 13.8125 20.001 13.6003 20.001H10.4003C10.1881 20.001 9.98464 19.9167 9.83461 19.7667C9.68458 19.6167 9.6003 19.4132 9.6003 19.201V18.401H14.4003V19.201Z" fill="url(#paint0_linear_691_82122)"/>
<path d="M16.0721 5.45703C15.3363 4.8519 14.4747 4.41865 13.5502 4.18891C12.6256 3.95916 11.6615 3.9387 10.7281 4.12903C9.48515 4.38009 8.34484 4.99469 7.4518 5.89486C6.55875 6.79502 5.95323 7.94018 5.71205 9.18503C5.53593 10.1184 5.56793 11.0792 5.80578 11.9987C6.04363 12.9183 6.48146 13.7741 7.08805 14.505C7.651 15.1403 7.97368 15.9526 8.00005 16.801V19.201C8.00005 19.8376 8.25291 20.448 8.703 20.8981C9.15308 21.3482 9.76353 21.601 10.4001 21.601H13.6001C14.2366 21.601 14.847 21.3482 15.2971 20.8981C15.7472 20.448 16.0001 19.8376 16.0001 19.201V16.953C16.0269 16.0164 16.371 15.1166 16.9761 14.401C18.0361 13.0897 18.5359 11.4129 18.3666 9.73513C18.1973 8.05738 17.3727 6.51422 16.0721 5.44103V5.45703ZM14.4001 19.201C14.4001 19.4132 14.3158 19.6167 14.1657 19.7667C14.0157 19.9167 13.8122 20.001 13.6001 20.001H10.4001C10.1879 20.001 9.9844 19.9167 9.83437 19.7667C9.68434 19.6167 9.60005 19.4132 9.60005 19.201V18.401H14.4001V19.201ZM15.7361 13.409C14.9317 14.3631 14.4623 15.5547 14.4001 16.801H12.8001V14.401C12.8001 14.1889 12.7158 13.9854 12.5657 13.8353C12.4157 13.6853 12.2122 13.601 12.0001 13.601C11.7879 13.601 11.5844 13.6853 11.4344 13.8353C11.2843 13.9854 11.2001 14.1889 11.2001 14.401V16.801H9.60005C9.57894 15.5755 9.12561 14.3968 8.32005 13.473C7.78856 12.8362 7.43116 12.0726 7.2827 11.2565C7.13423 10.4405 7.19977 9.59984 7.47292 8.81666C7.74607 8.03347 8.21751 7.33443 8.84127 6.78769C9.46503 6.24096 10.2198 5.86519 11.0321 5.69703C11.7301 5.55331 12.4514 5.56683 13.1435 5.73661C13.8357 5.90639 14.4814 6.22817 15.0337 6.67859C15.586 7.12902 16.0311 7.69677 16.3366 8.34065C16.6421 8.98452 16.8004 9.68835 16.8001 10.401C16.806 11.4969 16.4297 12.5606 15.7361 13.409Z" fill="#191C1D"/>
<rect x="4.61646" y="2.2019" width="3" height="1.5" rx="0.75" transform="rotate(45 4.61646 2.2019)" fill="#191C1D"/>
<rect x="1.30005" y="8.2019" width="3" height="1.5" rx="0.75" fill="#191C1D"/>
<rect x="19.429" y="8.2019" width="3" height="1.5" rx="0.75" fill="#191C1D"/>
<rect x="11.429" y="3" width="3" height="1.5" rx="0.75" transform="rotate(-90 11.429 3)" fill="#191C1D"/>
<rect width="3" height="1.5" rx="0.75" transform="matrix(0.707107 -0.707107 -0.707107 -0.707107 18.7126 5.2019)" fill="#191C1D"/>
<defs>
<linearGradient id="paint0_linear_691_82122" x1="5.60034" y1="12.8005" x2="18.3992" y2="12.8005" gradientUnits="userSpaceOnUse">
<stop stop-color="#FF930F"/>
<stop offset="1" stop-color="#FFF95B"/>
</linearGradient>
</defs>
</svg>

After

Width:  |  Height:  |  Size: 3.5 KiB

View file

@ -1,4 +1,4 @@
import type { UserAssetsResponse, AllowedUploadMimeType } from '@logto/schemas';
import type { AllowedUploadMimeType, UserAssets } from '@logto/schemas';
import { maxUploadFileSize } from '@logto/schemas';
import classNames from 'classnames';
import { useCallback, useState } from 'react';
@ -66,9 +66,7 @@ const FileUploader = ({ maxSize, allowedMimeTypes, limitDescription, onCompleted
try {
setIsUploading(true);
const { url } = await api
.post('api/user-assets', { body: formData })
.json<UserAssetsResponse>();
const { url } = await api.post('api/user-assets', { body: formData }).json<UserAssets>();
onCompleted(url);
} catch {

View file

@ -1,9 +1,10 @@
import type { AdminConsoleKey } from '@logto/phrases';
import classNames from 'classnames';
import type { KeyboardEventHandler, ReactNode } from 'react';
import type { KeyboardEventHandler, ReactElement, ReactNode } from 'react';
import { useCallback } from 'react';
import { useTranslation } from 'react-i18next';
import type DangerousRaw from '../DangerousRaw';
import * as styles from './Radio.module.scss';
const Check = () => (
@ -18,7 +19,7 @@ const Check = () => (
export type Props = {
className?: string;
value: string;
title?: AdminConsoleKey;
title?: AdminConsoleKey | ReactElement<typeof DangerousRaw>;
name?: string;
children?: ReactNode;
isChecked?: boolean;
@ -85,7 +86,7 @@ const Radio = ({
{children}
{type === 'plain' && <div className={styles.indicator} />}
{icon && <span className={styles.icon}>{icon}</span>}
{title && t(title)}
{title && (typeof title === 'string' ? t(title) : title)}
{isDisabled && disabledLabel && (
<div className={classNames(styles.indicator, styles.disabledLabel)}>
{t(disabledLabel)}

View file

@ -0,0 +1,12 @@
import type { UserAssetsServiceStatus } from '@logto/schemas';
import useSWRImmutable from 'swr/immutable';
const useUserAssetsService = () => {
const { data } = useSWRImmutable<UserAssetsServiceStatus>('api/user-assets/service-status');
return {
isReady: data?.status === 'ready',
};
};
export default useUserAssetsService;

View file

@ -1,11 +1,11 @@
import RadioGroup, { Radio } from '@/components/RadioGroup';
import type { Option } from './types';
import type { CardSelectorOption } from './types';
type Props = {
name: string;
value: string;
options: Option[];
options: CardSelectorOption[];
onChange: (value: string) => void;
optionClassName?: string;
};

View file

@ -29,6 +29,17 @@
}
}
.content {
.tag {
font: var(--font-body-3);
color: var(--color-text-secondary);
}
.trailingTag {
margin-left: _.unit(1);
}
}
&.selected {
border-color: var(--color-primary);
color: var(--color-text-link);

View file

@ -3,11 +3,11 @@ import { useTranslation } from 'react-i18next';
import { onKeyDownHandler } from '@/utils/a11y';
import type { Option } from '../types';
import type { MultiCardSelectorOption } from '../types';
import * as styles from './index.module.scss';
type Props = {
options: Option[];
options: MultiCardSelectorOption[];
value: string[];
onChange: (value: string[]) => void;
isNotAllowEmpty?: boolean;
@ -39,7 +39,7 @@ const MultiCardSelector = ({
return (
<div className={classNames(styles.selector, className)}>
{options.map(({ icon, title, value }) => (
{options.map(({ icon, title, value, tag, trailingTag }) => (
<div
key={value}
role="button"
@ -57,7 +57,15 @@ const MultiCardSelector = ({
})}
>
{icon && <span className={styles.icon}>{icon}</span>}
{t(title)}
<div className={styles.content}>
<div>
{typeof title === 'string' ? t(title) : title}
{trailingTag && (
<span className={classNames(styles.tag, styles.trailingTag)}>{t(trailingTag)}</span>
)}
</div>
{tag && <span className={styles.tag}>{t(tag)}</span>}
</div>
</div>
))}
</div>

View file

@ -1,3 +1,3 @@
export type { Option } from './types';
export type { CardSelectorOption, MultiCardSelectorOption } from './types';
export { default as CardSelector } from './CardSelector';
export { default as MultiCardSelector } from './MultiCardSelector';

View file

@ -1,8 +1,15 @@
import type { AdminConsoleKey } from '@logto/phrases';
import type { ReactNode } from 'react';
import type { ReactElement, ReactNode } from 'react';
export type Option = {
import type DangerousRaw from '@/components/DangerousRaw';
export type CardSelectorOption = {
icon?: ReactNode;
title: AdminConsoleKey;
title: AdminConsoleKey | ReactElement<typeof DangerousRaw>;
value: string;
};
export type MultiCardSelectorOption = CardSelectorOption & {
tag?: AdminConsoleKey;
trailingTag?: AdminConsoleKey;
};

View file

@ -1,14 +1,3 @@
import { SignInIdentifier } from '@logto/schemas';
import type { OnboardingSieConfig } from '../types';
import { Authentication } from '../types';
export const reservationLink = 'https://calendly.com/logto/30min';
export const logtoBlogLink =
'https://www.notion.so/silverhand/About-Logto-Cloud-Preview-ca316bc05f2a4b9188047da014124434?pvs=4';
export const defaultOnboardingSieConfig: OnboardingSieConfig = {
color: '#5D34F2',
identifier: SignInIdentifier.Email,
authentications: [Authentication.Password],
};

View file

@ -1,8 +1,11 @@
import type { Option as SelectorOption } from '@/onboarding/components/CardSelector';
import type {
CardSelectorOption,
MultiCardSelectorOption,
} from '@/onboarding/components/CardSelector';
import { CompanySize, Reason, Title } from '../../types';
export const titleOptions: SelectorOption[] = [
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 },
@ -11,7 +14,7 @@ export const titleOptions: SelectorOption[] = [
{ title: 'cloud.about.title_options.others', value: Title.Others },
];
export const companySizeOptions: SelectorOption[] = [
export const companySizeOptions: CardSelectorOption[] = [
{ title: 'cloud.about.company_options.size_1', value: CompanySize.Scale1 },
{ title: 'cloud.about.company_options.size_1_49', value: CompanySize.Scale2 },
{ title: 'cloud.about.company_options.size_50_199', value: CompanySize.Scale3 },
@ -19,7 +22,7 @@ export const companySizeOptions: SelectorOption[] = [
{ title: 'cloud.about.company_options.size_1000_plus', value: CompanySize.Scale5 },
];
export const reasonOptions: SelectorOption[] = [
export const reasonOptions: MultiCardSelectorOption[] = [
{ title: 'cloud.about.reason_options.adoption', value: Reason.Adoption },
{ title: 'cloud.about.reason_options.replacement', value: Reason.Replacement },
{ title: 'cloud.about.reason_options.evaluation', value: Reason.Evaluation },

View file

@ -0,0 +1,31 @@
@use '@/scss/underscore' as _;
.inspire {
margin-top: _.unit(3);
display: flex;
padding: _.unit(4) _.unit(5);
border-radius: 12px;
background-color: var(--color-base);
align-items: center;
.inspireContent {
margin-right: _.unit(6);
display: flex;
flex-direction: column;
.inspireTitle {
font: var(--font-title-2);
margin-bottom: _.unit(1);
}
.inspireDescription {
font: var(--font-body-2);
}
}
.button {
&:not(:disabled):not(:active):hover {
background: var(--color-layer-1);
}
}
}

View file

@ -0,0 +1,61 @@
import { ConnectorType } from '@logto/connector-kit';
import { useState } from 'react';
import { useTranslation } from 'react-i18next';
import Bulb from '@/assets/images/bulb.svg';
import LightBulb from '@/assets/images/light-bulb.svg';
import Button from '@/components/Button';
import useConnectorGroups from '@/hooks/use-connector-groups';
import type { OnboardingSieConfig } from '@/onboarding/types';
import { randomSieConfigTemplate } from '../../sie-config-templates';
import * as styles from './index.module.scss';
type Props = {
onInspired: (template: OnboardingSieConfig) => void;
};
const InspireMe = ({ onInspired }: Props) => {
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
const [isButtonHover, setIsButtonHover] = useState(false);
const BulbIcon = isButtonHover ? LightBulb : Bulb;
const [lastTemplateIndex, setLastTemplateIndex] = useState<number>();
const { data: connectorData, error } = useConnectorGroups();
const availableSocialTargets = error
? []
: connectorData
?.filter(({ type }) => type === ConnectorType.Social)
.map(({ target }) => target) ?? [];
const handleInspire = () => {
const { template, templateIndex } = randomSieConfigTemplate(
lastTemplateIndex,
availableSocialTargets
);
setLastTemplateIndex(templateIndex);
onInspired(template);
};
return (
<div className={styles.inspire}>
<div className={styles.inspireContent}>
<div className={styles.inspireTitle}>{t('cloud.sie.inspire.title')}</div>
<div className={styles.inspireDescription}>{t('cloud.sie.inspire.description')}</div>
</div>
<Button
icon={<BulbIcon />}
className={styles.button}
title="cloud.sie.inspire.inspire_me"
onMouseEnter={() => {
setIsButtonHover(true);
}}
onMouseLeave={() => {
setIsButtonHover(false);
}}
onClick={handleInspire}
/>
</div>
);
};
export default InspireMe;

View file

@ -0,0 +1,40 @@
import { ConnectorType } from '@logto/schemas';
import ConnectorLogo from '@/components/ConnectorLogo';
import DangerousRaw from '@/components/DangerousRaw';
import UnnamedTrans from '@/components/UnnamedTrans';
import useConnectorGroups from '@/hooks/use-connector-groups';
import type { MultiCardSelectorOption } from '@/onboarding/components/CardSelector';
import { MultiCardSelector } from '@/onboarding/components/CardSelector';
type Props = {
value: string[];
onChange: (value: string[]) => void;
};
const SocialSelector = ({ value, onChange }: Props) => {
const { data: connectorData, error } = useConnectorGroups();
if (!connectorData || error) {
return null;
}
const connectorOptions: MultiCardSelectorOption[] = connectorData
.filter(({ type }) => type === ConnectorType.Social)
.map((item) => {
return {
icon: <ConnectorLogo size="small" data={item} />,
title: (
<DangerousRaw>
<UnnamedTrans resource={item.name} />
</DangerousRaw>
),
value: item.target,
tag: 'general.trial',
};
});
return <MultiCardSelector options={connectorOptions} value={value} onChange={onChange} />;
};
export default SocialSelector;

View file

@ -2,52 +2,20 @@
.content {
display: flex;
flex: 1;
overflow-y: hidden;
padding: 0 _.unit(17);
.configWrapper {
flex: 1;
overflow: auto;
margin-right: _.unit(4);
padding: 0 _.unit(4) _.unit(6) 0;
}
.config {
background-color: var(--color-layer-1);
border-radius: 8px;
padding: _.unit(12);
min-width: 430px;
margin-right: _.unit(8);
.title {
margin-top: _.unit(6);
font: var(--font-title-1);
}
.inspire {
margin-top: _.unit(3);
display: flex;
padding: _.unit(4) _.unit(5);
border-radius: 12px;
background-color: var(--color-base);
align-items: center;
.inspireContent {
margin-right: _.unit(6);
display: flex;
flex-direction: column;
.inspireTitle {
font: var(--font-title-2);
margin-bottom: _.unit(1);
}
.inspireDescription {
font: var(--font-body-2);
}
}
}
.cardFieldHeadline {
margin-bottom: _.unit(2);
}
@ -59,8 +27,9 @@
.preview {
flex: 1;
margin-bottom: _.unit(6);
min-height: 580px;
position: sticky;
top: 0;
align-self: flex-start;
}
}

View file

@ -1,30 +1,36 @@
import type { SignInExperience as SignInExperienceType } from '@logto/schemas';
import { useEffect, useMemo } from 'react';
import { SignInIdentifier } from '@logto/schemas';
import { useCallback, useEffect, useMemo } from 'react';
import { Controller, useForm } from 'react-hook-form';
import { toast } from 'react-hot-toast';
import { useTranslation } from 'react-i18next';
import { useNavigate } from 'react-router-dom';
import useSWR from 'swr';
import Bulb from '@/assets/images/bulb.svg';
import Tools from '@/assets/images/tools.svg';
import Button from '@/components/Button';
import ColorPicker from '@/components/ColorPicker';
import FormField from '@/components/FormField';
import ImageUploader from '@/components/ImageUploader';
import OverlayScrollbar from '@/components/OverlayScrollbar';
import TextInput from '@/components/TextInput';
import type { RequestError } from '@/hooks/use-api';
import useApi from '@/hooks/use-api';
import useUserAssetsService from '@/hooks/use-user-assets-service';
import ActionBar from '@/onboarding/components/ActionBar';
import { CardSelector, MultiCardSelector } from '@/onboarding/components/CardSelector';
import { defaultOnboardingSieConfig } from '@/onboarding/constants';
import * as pageLayout from '@/onboarding/scss/layout.module.scss';
import { OnboardingPage } from '@/onboarding/types';
import { Authentication, OnboardingPage } from '@/onboarding/types';
import type { OnboardingSieConfig } from '@/onboarding/types';
import { getOnboardingPage } from '@/onboarding/utils';
import { uriValidator } from '@/utils/validator';
import InspireMe from './components/InspireMe';
import Preview from './components/Preview';
import SocialSelector from './components/SocialSelector';
import * as styles from './index.module.scss';
import { authenticationOptions, identifierOptions } from './options';
import { defaultOnboardingSieConfig } from './sie-config-templates';
import { parser } from './utils';
const SignInExperience = () => {
@ -33,21 +39,35 @@ const SignInExperience = () => {
const { data: signInExperience, mutate } = useSWR<SignInExperienceType, RequestError>(
'api/sign-in-exp'
);
const api = useApi();
const { isReady: isUserAssetsServiceReady } = useUserAssetsService();
const {
reset,
control,
watch,
register,
handleSubmit,
formState: { isSubmitting, isDirty },
getValues,
setValue,
formState: { isSubmitting, isDirty, errors },
} = useForm<OnboardingSieConfig>({ defaultValues: defaultOnboardingSieConfig });
const updateAuthenticationConfig = useCallback(() => {
const identifier = getValues('identifier');
if (identifier === SignInIdentifier.Username) {
setValue('authentications', [Authentication.Password]);
}
}, [getValues, setValue]);
useEffect(() => {
if (signInExperience) {
reset(parser.signInExperienceToOnboardSieConfig(signInExperience));
updateAuthenticationConfig();
}
}, [reset, signInExperience]);
}, [reset, signInExperience, updateAuthenticationConfig]);
const onboardingSieConfig = watch();
@ -57,13 +77,7 @@ const SignInExperience = () => {
}
}, [onboardingSieConfig, signInExperience]);
const handleInspireMe = () => {
// TODO @xiaoyijun
reset(defaultOnboardingSieConfig);
console.log('on inspire me');
};
const onSubmit = handleSubmit(async (formData) => {
const submit = (onSuccess: () => void) => async (formData: OnboardingSieConfig) => {
if (!signInExperience) {
return;
}
@ -75,46 +89,44 @@ const SignInExperience = () => {
.json<SignInExperienceType>();
void mutate(updatedData);
});
const handleBack = () => {
navigate(getOnboardingPage(OnboardingPage.AboutUser), { replace: true });
};
const handleSave = async () => {
await onSubmit();
toast.success(t('general.saved'));
};
const handleNext = async () => {
await onSubmit();
navigate(getOnboardingPage(OnboardingPage.Congrats), { replace: true });
onSuccess();
};
return (
<div className={pageLayout.page}>
<div className={styles.content}>
<OverlayScrollbar
options={{ scrollbars: { autoHide: 'scroll', autoHideDelay: 500 } }}
className={styles.configWrapper}
>
<OverlayScrollbar className={pageLayout.contentContainer}>
<div className={styles.content}>
<div className={styles.config}>
<Tools />
<div className={styles.title}>{t('cloud.sie.title')}</div>
<div className={styles.inspire}>
<div className={styles.inspireContent}>
<div className={styles.inspireTitle}>{t('cloud.sie.inspire.title')}</div>
<div className={styles.inspireDescription}>
{t('cloud.sie.inspire.description')}
</div>
</div>
<Button
icon={<Bulb />}
title="cloud.sie.inspire.inspire_me"
onClick={handleInspireMe}
/>
</div>
<FormField title="cloud.sie.logo_field">TBD</FormField>
<InspireMe
onInspired={(template) => {
reset(template);
updateAuthenticationConfig();
}}
/>
<FormField title="cloud.sie.logo_field">
{isUserAssetsServiceReady ? (
<Controller
name="logo"
control={control}
render={({ field: { onChange, value, name } }) => (
<ImageUploader name={name} value={value ?? ''} onChange={onChange} />
)}
/>
) : (
<TextInput
size="large"
{...register('logo', {
validate: (value) =>
!value || uriValidator(value) || t('errors.invalid_uri_format'),
})}
hasError={Boolean(errors.logo)}
errorMessage={errors.logo?.message}
/>
)}
</FormField>
<FormField title="cloud.sie.color_field">
<Controller
name="color"
@ -136,7 +148,10 @@ const SignInExperience = () => {
name={name}
value={value}
options={identifierOptions}
onChange={onChange}
onChange={(value) => {
onChange(value);
updateAuthenticationConfig();
}}
/>
)}
/>
@ -155,7 +170,11 @@ const SignInExperience = () => {
isNotAllowEmpty
className={styles.authnSelector}
value={value}
options={authenticationOptions}
options={authenticationOptions.filter(
({ value }) =>
onboardingSieConfig.identifier !== SignInIdentifier.Username ||
value === Authentication.Password
)}
onChange={onChange}
/>
)}
@ -166,16 +185,23 @@ const SignInExperience = () => {
title="cloud.sie.social_field"
headlineClassName={styles.cardFieldHeadline}
>
TBD
<Controller
name="socialTargets"
control={control}
defaultValue={defaultOnboardingSieConfig.socialTargets}
render={({ field: { value, onChange } }) => (
<SocialSelector value={value ?? []} onChange={onChange} />
)}
/>
</FormField>
</div>
</OverlayScrollbar>
<Preview
className={styles.preview}
signInExperience={previewSieConfig}
isLivePreviewDisabled={isDirty}
/>
</div>
<Preview
className={styles.preview}
signInExperience={previewSieConfig}
isLivePreviewDisabled={isDirty}
/>
</div>
</OverlayScrollbar>
<ActionBar step={3}>
<div className={styles.continueActions}>
@ -183,16 +209,34 @@ const SignInExperience = () => {
type="outline"
title="general.save"
disabled={isSubmitting}
onClick={handleSave}
onClick={async () => {
await handleSubmit(
submit(() => {
toast.success(t('general.saved'));
})
)();
}}
/>
<Button
type="primary"
title="cloud.sie.finish_and_done"
disabled={isSubmitting}
onClick={handleNext}
onClick={async () => {
await handleSubmit(
submit(() => {
navigate(getOnboardingPage(OnboardingPage.Congrats), { replace: true });
})
)();
}}
/>
</div>
<Button title="general.back" disabled={isSubmitting} onClick={handleBack} />
<Button
title="general.back"
disabled={isSubmitting}
onClick={() => {
navigate(getOnboardingPage(OnboardingPage.AboutUser), { replace: true });
}}
/>
</ActionBar>
</div>
);

View file

@ -5,10 +5,13 @@ import Keyboard from '@/assets/images/keyboard.svg';
import Label from '@/assets/images/label.svg';
import Lock from '@/assets/images/lock.svg';
import Mobile from '@/assets/images/mobile.svg';
import type { Option as SelectorOption } from '@/onboarding/components/CardSelector';
import type {
MultiCardSelectorOption,
CardSelectorOption,
} from '@/onboarding/components/CardSelector';
import { Authentication } from '@/onboarding/types';
export const identifierOptions: SelectorOption[] = [
export const identifierOptions: CardSelectorOption[] = [
{
icon: <Envelop />,
title: 'sign_in_exp.sign_up_and_sign_in.identifiers_email',
@ -26,7 +29,7 @@ export const identifierOptions: SelectorOption[] = [
},
];
export const authenticationOptions: SelectorOption[] = [
export const authenticationOptions: MultiCardSelectorOption[] = [
{
icon: <Lock />,
title: 'sign_in_exp.sign_up_and_sign_in.sign_in.password_auth',
@ -36,5 +39,6 @@ export const authenticationOptions: SelectorOption[] = [
icon: <Keyboard />,
title: 'sign_in_exp.sign_up_and_sign_in.sign_in.verification_code_auth',
value: Authentication.VerificationCode,
trailingTag: 'general.cap_limit',
},
];

View file

@ -0,0 +1,105 @@
import { SignInIdentifier } from '@logto/schemas';
import type { OnboardingSieConfig } from '@/onboarding/types';
import { Authentication } from '@/onboarding/types';
const assetsUrl =
'https://logtodev.blob.core.windows.net/public-blobs/admin/BY4BCq8GvfBF/2023/03/10';
export const defaultOnboardingSieConfig: OnboardingSieConfig = {
color: '#5D34F2',
identifier: SignInIdentifier.Email,
authentications: [Authentication.Password],
};
const configTemplate1: OnboardingSieConfig = {
logo: `${assetsUrl}/tVCAHjAB/logo1.png`,
color: '#19BEFD',
identifier: SignInIdentifier.Email,
authentications: [Authentication.Password, Authentication.VerificationCode],
};
const configTemplate2: OnboardingSieConfig = {
logo: `${assetsUrl}/kvCN1Z2a/logo2.png`,
color: '#F47346',
identifier: SignInIdentifier.Phone,
authentications: [Authentication.Password, Authentication.VerificationCode],
};
const configTemplate3: OnboardingSieConfig = {
logo: `${assetsUrl}/IcI0snBP/logo3.png`,
color: '#FF5449',
identifier: SignInIdentifier.Username,
authentications: [Authentication.Password],
};
const configTemplate4: OnboardingSieConfig = {
logo: `${assetsUrl}/7UQyvuFc/logo4.png`,
color: '#CA4E96',
identifier: SignInIdentifier.Email,
authentications: [Authentication.Password],
};
const configTemplate5: OnboardingSieConfig = {
logo: `${assetsUrl}/zB2merH1/logo5.png`,
color: '#F07EFF',
identifier: SignInIdentifier.Email,
authentications: [Authentication.VerificationCode],
};
const configTemplate6: OnboardingSieConfig = {
logo: `${assetsUrl}/CX51jxXS/logo6.png`,
color: '#9E65F8',
identifier: SignInIdentifier.Phone,
authentications: [Authentication.VerificationCode],
};
const configTemplate7: OnboardingSieConfig = {
logo: `${assetsUrl}/uLoMzrlz/logo7.png`,
color: '#FF5449',
identifier: SignInIdentifier.Email,
authentications: [Authentication.Password, Authentication.VerificationCode],
};
const configTemplate8: OnboardingSieConfig = {
logo: `${assetsUrl}/dIz8UHEh/logo8.png`,
color: '#5D34F2',
identifier: SignInIdentifier.Phone,
authentications: [Authentication.VerificationCode],
};
const sieConfigTemplates: OnboardingSieConfig[] = [
configTemplate1,
configTemplate2,
configTemplate3,
configTemplate4,
configTemplate5,
configTemplate6,
configTemplate7,
configTemplate8,
];
export const randomSieConfigTemplate = (
lastTemplateIndex: number | undefined,
availableSocialTargets: string[]
) => {
// Get random template
const randomIndex = Math.floor(Math.random() * sieConfigTemplates.length);
const index =
randomIndex === lastTemplateIndex ? (randomIndex + 1) % sieConfigTemplates.length : randomIndex;
const template = sieConfigTemplates[index] ?? configTemplate1;
// Take the first two after shuffling.
const socialTargets = availableSocialTargets
.slice()
.sort(() => 0.5 - Math.random())
.slice(0, 2);
return {
template: {
...template,
socialTargets,
},
templateIndex: index,
};
};

View file

@ -1,5 +1,6 @@
import type { SignInExperience } from '@logto/schemas';
import { SignInIdentifier } from '@logto/schemas';
import { conditional } from '@silverhand/essentials';
import type { OnboardingSieConfig } from '@/onboarding/types';
import { Authentication } from '@/onboarding/types';
@ -9,8 +10,10 @@ const signInExperienceToOnboardSieConfig = (
): OnboardingSieConfig => {
const {
color: { primaryColor },
branding: { logoUrl: logo },
signIn: { methods: signInMethods },
signUp: { identifiers: signUpIdentifiers },
socialSignInConnectorTargets,
} = signInExperience;
const identifier =
@ -28,9 +31,11 @@ const signInExperienceToOnboardSieConfig = (
}, []);
return {
logo,
color: primaryColor,
identifier,
authentications,
socialTargets: socialSignInConnectorTargets,
};
};
@ -38,16 +43,19 @@ const onboardSieConfigToSignInExperience = (
config: OnboardingSieConfig,
basedConfig: SignInExperience
): SignInExperience => {
const { color: onboardConfigColor, identifier, authentications } = config;
const { color: baseColorConfig } = basedConfig;
const { logo, color: onboardConfigColor, identifier, authentications, socialTargets } = config;
const { color: baseColorConfig, branding: baseBranding } = basedConfig;
const isPasswordSetup = authentications.includes(Authentication.Password);
const isVerificationCodeSetup =
authentications.includes(Authentication.VerificationCode) &&
identifier !== SignInIdentifier.Username;
const isVerificationCodeSetup = identifier !== SignInIdentifier.Username;
const signInExperience = {
const signInExperience: SignInExperience = {
...basedConfig,
branding: {
...baseBranding,
logoUrl: conditional(logo?.length && logo),
darkLogoUrl: conditional(logo?.length && logo),
},
color: {
...baseColorConfig,
primaryColor: onboardConfigColor,
@ -67,6 +75,7 @@ const onboardSieConfigToSignInExperience = (
},
],
},
socialSignInConnectorTargets: socialTargets ?? [],
};
return signInExperience;

View file

@ -2,11 +2,11 @@ import Building from '@/assets/images/building.svg';
import Cloud from '@/assets/images/cloud.svg';
import Database from '@/assets/images/database.svg';
import Pizza from '@/assets/images/pizza.svg';
import type { Option as SelectorOption } from '@/onboarding/components/CardSelector';
import type { CardSelectorOption } from '@/onboarding/components/CardSelector';
import { DeploymentType, Project } from '../../types';
export const projectOptions: SelectorOption[] = [
export const projectOptions: CardSelectorOption[] = [
{
icon: <Pizza />,
title: 'cloud.welcome.project_options.personal',
@ -19,7 +19,7 @@ export const projectOptions: SelectorOption[] = [
},
];
export const deploymentTypeOptions: SelectorOption[] = [
export const deploymentTypeOptions: CardSelectorOption[] = [
{
icon: <Database />,
title: 'cloud.welcome.deployment_type_options.open_source',

View file

@ -72,7 +72,9 @@ export enum Authentication {
}
export type OnboardingSieConfig = {
logo?: string;
color: string;
identifier: SignInIdentifier;
authentications: Authentication[];
socialTargets?: string[];
};

View file

@ -1,7 +1,7 @@
import { readFile } from 'fs/promises';
import { generateStandardId } from '@logto/core-kit';
import type { UserAssetsResponse } from '@logto/schemas';
import type { UserAssets } from '@logto/schemas';
import {
userAssetsGuard,
userAssetsServiceStatusGuard,
@ -81,7 +81,7 @@ export default function userAssetsRoutes<T extends AuthedRouter>(...[router]: Ro
publicUrl: storageProviderConfig.publicUrl,
});
const result: UserAssetsResponse = {
const result: UserAssets = {
url,
};

View file

@ -104,6 +104,12 @@ describe('smoke testing for cloud', () => {
});
it('can complete the sie configuration process and enter the congrats page', async () => {
// Wait for the sie config to load
await page.waitForTimeout(1000);
// Select username as the identifier
await expect(page).toClick('div[role=radio]:has(input[name=identifier][value=username])');
// Click the finish button
await expect(page).toClick('div[class$=continueActions] button:last-child');

View file

@ -51,6 +51,7 @@ const general = {
try_now: 'Try Now', // UNTRANSLATED
multiple_form_field: '(Multiple)', // UNTRANSLATED
cap_limit: 'Cap limit', // UNTRANSLATED
trial: 'Trail', // UNTRANSLATED
};
export default general;

View file

@ -50,6 +50,7 @@ const general = {
try_now: 'Try Now',
multiple_form_field: '(Multiple)',
cap_limit: 'Cap limit',
trial: 'Trail',
};
export default general;

View file

@ -51,6 +51,7 @@ const general = {
try_now: 'Try Now', // UNTRANSLATED
multiple_form_field: '(Multiple)', // UNTRANSLATED
cap_limit: 'Cap limit', // UNTRANSLATED
trial: 'Trail', // UNTRANSLATED
};
export default general;

View file

@ -50,6 +50,7 @@ const general = {
try_now: 'Try Now', // UNTRANSLATED
multiple_form_field: '(Multiple)', // UNTRANSLATED
cap_limit: 'Cap limit', // UNTRANSLATED
trial: 'Trail', // UNTRANSLATED
};
export default general;

View file

@ -51,6 +51,7 @@ const general = {
try_now: 'Try Now', // UNTRANSLATED
multiple_form_field: '(Multiple)', // UNTRANSLATED
cap_limit: 'Cap limit', // UNTRANSLATED
trial: 'Trail', // UNTRANSLATED
};
export default general;

View file

@ -50,6 +50,7 @@ const general = {
try_now: 'Try Now', // UNTRANSLATED
multiple_form_field: '(Multiple)', // UNTRANSLATED
cap_limit: 'Cap limit', // UNTRANSLATED
trial: 'Trail', // UNTRANSLATED
};
export default general;

View file

@ -51,6 +51,7 @@ const general = {
try_now: 'Try Now', // UNTRANSLATED
multiple_form_field: '(Multiple)', // UNTRANSLATED
cap_limit: 'Cap limit', // UNTRANSLATED
trial: 'Trail', // UNTRANSLATED
};
export default general;

View file

@ -50,6 +50,7 @@ const general = {
try_now: 'Try Now', // UNTRANSLATED
multiple_form_field: '(Multiple)', // UNTRANSLATED
cap_limit: 'Cap limit', // UNTRANSLATED
trial: 'Trail', // UNTRANSLATED
};
export default general;

View file

@ -24,10 +24,10 @@ export const userAssetsServiceStatusGuard = z.object({
maxUploadFileSize: z.number().optional(),
});
export type UserAssetsServiceStatusResponse = z.infer<typeof userAssetsServiceStatusGuard>;
export type UserAssetsServiceStatus = z.infer<typeof userAssetsServiceStatusGuard>;
export const userAssetsGuard = z.object({
url: z.string(),
});
export type UserAssetsResponse = z.infer<typeof userAssetsGuard>;
export type UserAssets = z.infer<typeof userAssetsGuard>;