diff --git a/packages/console/public/get-started/react/en/index.json b/packages/console/public/get-started/application/react/en/index.json similarity index 100% rename from packages/console/public/get-started/react/en/index.json rename to packages/console/public/get-started/application/react/en/index.json diff --git a/packages/console/public/get-started/react/en/step1.md b/packages/console/public/get-started/application/react/en/step1.md similarity index 100% rename from packages/console/public/get-started/react/en/step1.md rename to packages/console/public/get-started/application/react/en/step1.md diff --git a/packages/console/public/get-started/react/en/step2.md b/packages/console/public/get-started/application/react/en/step2.md similarity index 100% rename from packages/console/public/get-started/react/en/step2.md rename to packages/console/public/get-started/application/react/en/step2.md diff --git a/packages/console/public/get-started/react/en/step3.md b/packages/console/public/get-started/application/react/en/step3.md similarity index 100% rename from packages/console/public/get-started/react/en/step3.md rename to packages/console/public/get-started/application/react/en/step3.md diff --git a/packages/console/public/get-started/react/en/step4.md b/packages/console/public/get-started/application/react/en/step4.md similarity index 100% rename from packages/console/public/get-started/react/en/step4.md rename to packages/console/public/get-started/application/react/en/step4.md diff --git a/packages/console/public/get-started/react/en/step5.md b/packages/console/public/get-started/application/react/en/step5.md similarity index 100% rename from packages/console/public/get-started/react/en/step5.md rename to packages/console/public/get-started/application/react/en/step5.md diff --git a/packages/console/public/get-started/react/zh-CN/index.json b/packages/console/public/get-started/application/react/zh-CN/index.json similarity index 100% rename from packages/console/public/get-started/react/zh-CN/index.json rename to packages/console/public/get-started/application/react/zh-CN/index.json diff --git a/packages/console/public/get-started/react/zh-CN/step1.md b/packages/console/public/get-started/application/react/zh-CN/step1.md similarity index 100% rename from packages/console/public/get-started/react/zh-CN/step1.md rename to packages/console/public/get-started/application/react/zh-CN/step1.md diff --git a/packages/console/public/get-started/react/zh-CN/step2.md b/packages/console/public/get-started/application/react/zh-CN/step2.md similarity index 100% rename from packages/console/public/get-started/react/zh-CN/step2.md rename to packages/console/public/get-started/application/react/zh-CN/step2.md diff --git a/packages/console/public/get-started/react/zh-CN/step3.md b/packages/console/public/get-started/application/react/zh-CN/step3.md similarity index 100% rename from packages/console/public/get-started/react/zh-CN/step3.md rename to packages/console/public/get-started/application/react/zh-CN/step3.md diff --git a/packages/console/public/get-started/react/zh-CN/step4.md b/packages/console/public/get-started/application/react/zh-CN/step4.md similarity index 100% rename from packages/console/public/get-started/react/zh-CN/step4.md rename to packages/console/public/get-started/application/react/zh-CN/step4.md diff --git a/packages/console/public/get-started/react/zh-CN/step5.md b/packages/console/public/get-started/application/react/zh-CN/step5.md similarity index 100% rename from packages/console/public/get-started/react/zh-CN/step5.md rename to packages/console/public/get-started/application/react/zh-CN/step5.md diff --git a/packages/console/src/App.tsx b/packages/console/src/App.tsx index f26461180..e0e8bb9a9 100644 --- a/packages/console/src/App.tsx +++ b/packages/console/src/App.tsx @@ -17,7 +17,6 @@ import Applications from './pages/Applications'; import Callback from './pages/Callback'; import ConnectorDetails from './pages/ConnectorDetails'; import Connectors from './pages/Connectors'; -import GetStarted from './pages/GetStarted'; import NotFound from './pages/NotFound'; import UserDetails from './pages/UserDetails'; import Users from './pages/Users'; @@ -44,7 +43,6 @@ const Main = () => { } /> }> } /> - } /> } /> diff --git a/packages/console/src/components/Card/index.tsx b/packages/console/src/components/Card/index.tsx index 79275422f..684bd4df6 100644 --- a/packages/console/src/components/Card/index.tsx +++ b/packages/console/src/components/Card/index.tsx @@ -1,5 +1,5 @@ import classNames from 'classnames'; -import React, { forwardRef, LegacyRef, ReactNode } from 'react'; +import React, { forwardRef, Ref, ReactNode } from 'react'; import * as styles from './index.module.scss'; @@ -8,7 +8,7 @@ type Props = { className?: string; }; -const Card = (props: Props, reference?: LegacyRef) => { +const Card = (props: Props, reference?: Ref) => { const { children, className } = props; return ( diff --git a/packages/console/src/pages/Applications/components/CreateForm/index.tsx b/packages/console/src/pages/Applications/components/CreateForm/index.tsx index df47e190c..424fe8ad0 100644 --- a/packages/console/src/pages/Applications/components/CreateForm/index.tsx +++ b/packages/console/src/pages/Applications/components/CreateForm/index.tsx @@ -11,10 +11,11 @@ import ModalLayout from '@/components/ModalLayout'; import RadioGroup, { Radio } from '@/components/RadioGroup'; import TextInput from '@/components/TextInput'; import useApi, { RequestError } from '@/hooks/use-api'; +import GetStarted from '@/pages/GetStarted'; import * as modalStyles from '@/scss/modal.module.scss'; -import { applicationTypeI18nKey } from '@/types/applications'; +import { applicationTypeI18nKey, SupportedJavascriptLibraries } from '@/types/applications'; -import GetStarted from '../GetStarted'; +import LibrarySelector from '../LibrarySelector'; import TypeDescription from '../TypeDescription'; import * as styles from './index.module.scss'; @@ -108,7 +109,15 @@ const CreateForm = ({ onClose }: Props) => { {!isGetStartedSkipped && createdApp && ( - + } + title={createdApp.name} + subtitle="applications.get_started.header_description" + type="application" + defaultSubtype={SupportedJavascriptLibraries.React} + onClose={closeModal} + onComplete={closeModal} + /> )} diff --git a/packages/console/src/pages/Applications/components/GetStarted/index.module.scss b/packages/console/src/pages/Applications/components/GetStarted/index.module.scss deleted file mode 100644 index 925531ea8..000000000 --- a/packages/console/src/pages/Applications/components/GetStarted/index.module.scss +++ /dev/null @@ -1,134 +0,0 @@ -@use '@/scss/underscore' as _; - -.quickStartGuide { - display: flex; - flex-direction: column; - background-color: var(--color-main-background); - height: 100vh; - - .header { - display: flex; - align-items: center; - background-color: var(--color-on-primary); - height: 64px; - padding: 0 _.unit(21) 0 _.unit(2); - - button { - margin-left: _.unit(4); - } - - .separator { - @include _.vertical-bar; - height: 20px; - margin: 0 _.unit(5) 0 _.unit(4); - } - } - - .content { - flex: 1; - display: flex; - flex-direction: column; - align-items: center; - overflow-y: auto; - padding: _.unit(6) 0 _.unit(20); - } - - .card { - width: 858px; - padding: _.unit(5) _.unit(6); - display: flex; - flex-direction: column; - scroll-margin: _.unit(5); - - &.selector { - display: block; - - .title { - font: var(--font-title-large); - } - - .subtitle { - font: var(--font-body-medium); - color: var(--color-component-text); - margin-top: _.unit(3); - } - - .radioGroup { - margin-top: _.unit(6); - } - - .radio { - border-radius: _.unit(2); - width: 240px; - max-width: unset; - } - - &.folded { - display: flex; - flex-direction: row; - align-items: center; - flex: 0 0 56px; - background: var(--color-neutral-variant-90); - border-radius: _.unit(2); - padding: 0 _.unit(4); - font: var(--font-body-medium); - color: var(--color-component-text); - - img { - margin-right: _.unit(3); - } - } - } - - .cardHeader { - display: flex; - flex-direction: row; - align-items: center; - cursor: pointer; - - > svg { - fill: var(--color-icon); - } - - .index { - display: flex; - align-items: center; - justify-content: center; - width: 28px; - height: 28px; - border-radius: 50%; - color: var(--color-primary); - background: var(--color-surface-5); - font: var(--font-title-medium); - margin-right: _.unit(4); - - &.active { - color: var(--color-on-primary); - background: var(--color-primary); - } - - &.completed { - background: var(--color-primary); - - > svg { - fill: var(--color-on-primary); - } - } - } - } - - .buttonWrapper { - display: flex; - justify-content: flex-end; - margin-top: _.unit(6); - } - } - - .card + .card { - margin-top: _.unit(6); - } - - .markdownContent { - margin-top: _.unit(6); - } -} diff --git a/packages/console/src/pages/Applications/components/GetStarted/index.tsx b/packages/console/src/pages/Applications/components/GetStarted/index.tsx deleted file mode 100644 index 85a8f3a67..000000000 --- a/packages/console/src/pages/Applications/components/GetStarted/index.tsx +++ /dev/null @@ -1,215 +0,0 @@ -import classNames from 'classnames'; -import i18next from 'i18next'; -import React, { useEffect, useMemo, useState } from 'react'; -import { useTranslation } from 'react-i18next'; -import ReactMarkdown from 'react-markdown'; -// eslint-disable-next-line node/file-extension-in-import -import useSWRImmutable from 'swr/immutable'; - -import highFive from '@/assets/images/high-five.svg'; -import tada from '@/assets/images/tada.svg'; -import Button from '@/components/Button'; -import Card from '@/components/Card'; -import CardTitle from '@/components/CardTitle'; -import DangerousRaw from '@/components/DangerousRaw'; -import IconButton from '@/components/IconButton'; -import RadioGroup, { Radio } from '@/components/RadioGroup'; -import Spacer from '@/components/Spacer'; -import { ArrowDown, ArrowUp } from '@/icons/Arrow'; -import Close from '@/icons/Close'; -import Tick from '@/icons/Tick'; -import { SupportedJavascriptLibraries } from '@/types/applications'; -import { parseMarkdownWithYamlFrontmatter } from '@/utilities/markdown'; - -import * as styles from './index.module.scss'; - -type Props = { - appName: string; - onClose: () => void; -}; - -type DocumentFileNames = { - files: string[]; -}; - -type Step = { - title?: string; - subtitle?: string; - metadata: string; -}; - -const GetStarted = ({ appName, onClose }: Props) => { - const [isLibrarySelectorFolded, setIsLibrarySelectorFolded] = useState(false); - const [libraryName, setLibraryName] = useState(SupportedJavascriptLibraries.React); - const [activeStepIndex, setActiveStepIndex] = useState(-1); - const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' }); - const publicPath = useMemo( - () => `/console/get-started/${libraryName}/${i18next.language}`, - [libraryName] - ); - - const { data: jsonData } = useSWRImmutable(`${publicPath}/index.json`); - const { data: steps } = useSWRImmutable(jsonData, async ({ files }: DocumentFileNames) => - Promise.all( - files.map(async (fileName) => { - const response = await fetch(`${publicPath}/${fileName}`); - const markdownFile = await response.text(); - - return parseMarkdownWithYamlFrontmatter(markdownFile); - }) - ) - ); - - const stepReferences = useMemo( - () => Array.from({ length: steps?.length ?? 0 }).map(() => React.createRef()), - [steps?.length] - ); - - useEffect(() => { - if (activeStepIndex > -1) { - const activeStepReference = stepReferences[activeStepIndex]; - activeStepReference?.current?.scrollIntoView({ block: 'start', behavior: 'smooth' }); - } - }, [activeStepIndex, stepReferences]); - - const onClickFetchSampleProject = () => { - window.open( - `https://github.com/logto-io/js/tree/master/packages/${libraryName.toLowerCase()}-sample`, - '_blank' - ); - }; - - const librarySelector = useMemo( - () => ( - - success -
-
{t('applications.get_started.title')}
-
{t('applications.get_started.subtitle')}
-
- - {Object.values(SupportedJavascriptLibraries).map((library) => ( - - ))} - -
-
-
- ), - [libraryName, t] - ); - - const librarySelectorFolded = useMemo( - () => ( -
- Tada! - - {t('applications.get_started.description_by_library', { library: libraryName })} - -
- ), - [libraryName, t] - ); - - return ( -
-
- - - -
- {appName}} - subtitle="applications.get_started.header_description" - /> - -
-
- {isLibrarySelectorFolded ? librarySelectorFolded : librarySelector} - {steps?.map((step, index) => { - const { title, subtitle, metadata } = step; - const isExpanded = activeStepIndex === index; - const isCompleted = activeStepIndex > index; - const isLastStep = index === steps.length - 1; - - // Steps in get-started must have "title" declared in the Yaml header of the markdown source file - if (!title) { - return null; - } - - // TODO: add more styles to markdown renderer - // TODO: render form and input fields in steps - return ( - -
{ - setIsLibrarySelectorFolded(true); - setActiveStepIndex(index); - }} - > -
- {isCompleted ? : index + 1} -
- {title}} - subtitle={{subtitle}} - /> - - {isExpanded ? : } -
- {isExpanded && ( - <> - {metadata} -
-
- - )} -
- ); - })} -
-
- ); -}; - -export default GetStarted; diff --git a/packages/console/src/pages/Applications/components/LibrarySelector/index.module.scss b/packages/console/src/pages/Applications/components/LibrarySelector/index.module.scss new file mode 100644 index 000000000..6e1e59dcd --- /dev/null +++ b/packages/console/src/pages/Applications/components/LibrarySelector/index.module.scss @@ -0,0 +1,50 @@ +@use '@/scss/underscore' as _; + +.card { + padding: _.unit(5) _.unit(6); + display: block; + flex-direction: column; + scroll-margin: _.unit(5); + + .title { + font: var(--font-title-large); + } + + .subtitle { + font: var(--font-body-medium); + color: var(--color-component-text); + margin-top: _.unit(3); + } + + .radioGroup { + margin-top: _.unit(6); + } + + .radio { + border-radius: _.unit(2); + width: 240px; + max-width: unset; + } + + &.folded { + display: flex; + flex-direction: row; + align-items: center; + flex: 0 0 56px; + background: var(--color-neutral-variant-90); + border-radius: _.unit(2); + padding: 0 _.unit(4); + font: var(--font-body-medium); + color: var(--color-component-text); + + img { + margin-right: _.unit(3); + } + } + + .buttonWrapper { + display: flex; + justify-content: flex-end; + margin-top: _.unit(6); + } +} diff --git a/packages/console/src/pages/Applications/components/LibrarySelector/index.tsx b/packages/console/src/pages/Applications/components/LibrarySelector/index.tsx new file mode 100644 index 000000000..35a6ed6c1 --- /dev/null +++ b/packages/console/src/pages/Applications/components/LibrarySelector/index.tsx @@ -0,0 +1,78 @@ +import classNames from 'classnames'; +import React, { useState, useMemo } from 'react'; +import { useTranslation } from 'react-i18next'; + +import highFive from '@/assets/images/high-five.svg'; +import tada from '@/assets/images/tada.svg'; +import Button from '@/components/Button'; +import Card from '@/components/Card'; +import RadioGroup, { Radio } from '@/components/RadioGroup'; +import { SupportedJavascriptLibraries } from '@/types/applications'; + +import * as styles from './index.module.scss'; + +type Props = { + className?: string; + libraryName?: SupportedJavascriptLibraries; + onChange?: (value: string) => void; + onToggle?: () => void; +}; + +const LibrarySelector = ({ + className, + libraryName = SupportedJavascriptLibraries.React, + onChange, + onToggle, +}: Props) => { + const [isFolded, setIsFolded] = useState(false); + const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' }); + + const librarySelector = useMemo( + () => ( + + success +
+
{t('applications.get_started.title')}
+
{t('applications.get_started.subtitle')}
+
+ + {Object.values(SupportedJavascriptLibraries).map((library) => ( + + ))} + +
+
+
+ ), + [className, libraryName, onChange, onToggle, t] + ); + + const librarySelectorFolded = useMemo( + () => ( +
+ Tada! + + {t('applications.get_started.description_by_library', { library: libraryName })} + +
+ ), + [className, libraryName, t] + ); + + return isFolded ? librarySelectorFolded : librarySelector; +}; + +export default LibrarySelector; diff --git a/packages/console/src/pages/GetStarted/components/Step/index.module.scss b/packages/console/src/pages/GetStarted/components/Step/index.module.scss new file mode 100644 index 000000000..78867a8ef --- /dev/null +++ b/packages/console/src/pages/GetStarted/components/Step/index.module.scss @@ -0,0 +1,59 @@ +@use '@/scss/underscore' as _; + +.card { + padding: _.unit(5) _.unit(6); + display: flex; + flex-direction: column; + scroll-margin: _.unit(5); + + .cardHeader { + display: flex; + flex-direction: row; + align-items: center; + cursor: pointer; + + > svg { + fill: var(--color-icon); + } + + .index { + display: flex; + align-items: center; + justify-content: center; + width: 28px; + height: 28px; + border-radius: 50%; + color: var(--color-primary); + background: var(--color-surface-5); + font: var(--font-title-medium); + margin-right: _.unit(4); + + &.active { + color: var(--color-on-primary); + background: var(--color-primary); + } + + &.completed { + background: var(--color-primary); + + > svg { + fill: var(--color-on-primary); + } + } + } + } + + .buttonWrapper { + display: flex; + justify-content: flex-end; + margin-top: _.unit(6); + } +} + +.card + .card { + margin-top: _.unit(6); +} + +.markdownContent { + margin-top: _.unit(6); +} diff --git a/packages/console/src/pages/GetStarted/components/Step/index.tsx b/packages/console/src/pages/GetStarted/components/Step/index.tsx new file mode 100644 index 000000000..3d29aea64 --- /dev/null +++ b/packages/console/src/pages/GetStarted/components/Step/index.tsx @@ -0,0 +1,88 @@ +import classNames from 'classnames'; +import React, { forwardRef, Ref } from 'react'; +import ReactMarkdown from 'react-markdown'; + +import Button from '@/components/Button'; +import Card from '@/components/Card'; +import CardTitle from '@/components/CardTitle'; +import DangerousRaw from '@/components/DangerousRaw'; +import IconButton from '@/components/IconButton'; +import Spacer from '@/components/Spacer'; +import { ArrowDown, ArrowUp } from '@/icons/Arrow'; +import Tick from '@/icons/Tick'; + +import * as styles from './index.module.scss'; + +export type StepMetadata = { + title?: string; + subtitle?: string; + metadata: string; // Markdown formatted string +}; + +type Props = { + data: StepMetadata; + index: number; + isCompleted: boolean; + isExpanded: boolean; + isFinalStep: boolean; + onComplete?: () => void; + onNext?: () => void; + onToggle?: () => void; +}; + +const Step = ( + { data, index, isCompleted, isExpanded, isFinalStep, onComplete, onNext, onToggle }: Props, + ref?: Ref +) => { + const { title, subtitle, metadata } = data; + + // Steps in get-started must have "title" declared in the Yaml header of the markdown source file + if (!title) { + return null; + } + + // TODO: add more styles to markdown renderer + // TODO: render form and input fields in steps + return ( + +
+
+ {isCompleted ? : index + 1} +
+ {title}} + subtitle={{subtitle}} + /> + + {isExpanded ? : } +
+ {isExpanded && ( + <> + {metadata} +
+
+ + )} +
+ ); +}; + +export default forwardRef(Step); diff --git a/packages/console/src/pages/GetStarted/hooks/index.ts b/packages/console/src/pages/GetStarted/hooks/index.ts new file mode 100644 index 000000000..4de36229c --- /dev/null +++ b/packages/console/src/pages/GetStarted/hooks/index.ts @@ -0,0 +1,44 @@ +import i18next from 'i18next'; +import { useMemo } from 'react'; +// eslint-disable-next-line node/file-extension-in-import +import useSWRImmutable from 'swr/immutable'; + +import { parseMarkdownWithYamlFrontmatter } from '@/utilities/markdown'; + +import { StepMetadata } from '../components/Step'; + +type DocumentFileNames = { + files: string[]; +}; + +export type GetStartedType = 'application' | 'connector'; + +/** + * Fetch the markdown files for the given type and subtype. + * @param type 'application' or 'connector' + * @param subtype Application library name or connector name + * @returns List of step metadata including Yaml frontmatter and markdown content + */ +export const useGetStartedSteps = (type: GetStartedType, subtype?: string) => { + const subPath = subtype ? `/${subtype}` : ''; + const publicPath = useMemo( + () => `/console/get-started/${type}${subPath}/${i18next.language}`, + [type, subPath] + ); + + const { data: jsonData } = useSWRImmutable(`${publicPath}/index.json`); + const { data: steps } = useSWRImmutable( + jsonData, + async ({ files }: DocumentFileNames) => + Promise.all( + files.map(async (fileName) => { + const response = await fetch(`${publicPath}/${fileName}`); + const markdownFile = await response.text(); + + return parseMarkdownWithYamlFrontmatter(markdownFile); + }) + ) + ); + + return steps; +}; diff --git a/packages/console/src/pages/GetStarted/index.module.scss b/packages/console/src/pages/GetStarted/index.module.scss new file mode 100644 index 000000000..ddb7fcba4 --- /dev/null +++ b/packages/console/src/pages/GetStarted/index.module.scss @@ -0,0 +1,43 @@ +@use '@/scss/underscore' as _; + +.container { + display: flex; + flex-direction: column; + background-color: var(--color-main-background); + height: 100vh; + + .header { + display: flex; + align-items: center; + background-color: var(--color-on-primary); + height: 64px; + padding: 0 _.unit(21) 0 _.unit(2); + + button { + margin-left: _.unit(4); + } + + .separator { + @include _.vertical-bar; + height: 20px; + margin: 0 _.unit(5) 0 _.unit(4); + } + } + + .content { + flex: 1; + display: flex; + flex-direction: column; + align-items: center; + overflow-y: auto; + padding: _.unit(6) 0 _.unit(20); + + > * { + width: 858px; + } + + .banner { + margin-bottom: _.unit(6); + } + } +} diff --git a/packages/console/src/pages/GetStarted/index.tsx b/packages/console/src/pages/GetStarted/index.tsx index f667aeece..fc891763e 100644 --- a/packages/console/src/pages/GetStarted/index.tsx +++ b/packages/console/src/pages/GetStarted/index.tsx @@ -1,7 +1,129 @@ -import React from 'react'; +import { AdminConsoleKey } from '@logto/phrases'; +import { Nullable } from '@silverhand/essentials'; +import React, { + cloneElement, + isValidElement, + PropsWithChildren, + ReactNode, + useEffect, + useRef, + useState, +} from 'react'; -const GetStarted = () => { - return
GetStarted
; +import Button from '@/components/Button'; +import CardTitle from '@/components/CardTitle'; +import DangerousRaw from '@/components/DangerousRaw'; +import IconButton from '@/components/IconButton'; +import Spacer from '@/components/Spacer'; +import Close from '@/icons/Close'; + +import Step from './components/Step'; +import { GetStartedType, useGetStartedSteps } from './hooks'; +import * as styles from './index.module.scss'; + +type Props = PropsWithChildren<{ + title: string; + subtitle?: AdminConsoleKey; + type: GetStartedType; + /** `subtype` can be an actual type of an application or connector. + * e.g. React, Angular, Vue, etc. for application. Or Github, WeChat, etc. for connector. + */ + defaultSubtype?: string; + bannerComponent?: ReactNode; + onClose?: () => void; + onComplete?: () => void; + onToggleSteps?: () => void; +}>; + +const onClickFetchSampleProject = (projectName: string) => { + const sampleUrl = `https://github.com/logto-io/js/tree/master/packages/${projectName}-sample`; + window.open(sampleUrl, '_blank'); +}; + +const GetStarted = ({ + title, + subtitle, + type, + defaultSubtype, + bannerComponent, + onClose, + onComplete, + onToggleSteps, +}: Props) => { + const [subtype, setSubtype] = useState(defaultSubtype); + const [activeStepIndex, setActiveStepIndex] = useState(-1); + const steps = useGetStartedSteps(type, subtype) ?? []; + + const stepReferences = useRef>>( + // eslint-disable-next-line @typescript-eslint/ban-types + Array.from({ length: steps.length }).fill(null) + ); + + useEffect(() => { + if (activeStepIndex > -1) { + const activeStepRef = stepReferences.current[activeStepIndex]; + activeStepRef?.scrollIntoView({ block: 'start', behavior: 'smooth' }); + } + }, [activeStepIndex, stepReferences]); + + return ( +
+
+ + + +
+ {title}} subtitle={subtitle} /> + +
+
+ {isValidElement(bannerComponent) && + cloneElement(bannerComponent, { + className: styles.banner, + onChange: setSubtype, + onToggle: () => { + setActiveStepIndex(0); + }, + })} + {steps.map((step, index) => { + const isFinalStep = index === steps.length - 1; + + return ( + { + // eslint-disable-next-line @silverhand/fp/no-mutation + stepReferences.current[index] = element; + }} + data={step} + index={index} + isCompleted={activeStepIndex > index} + isExpanded={activeStepIndex === index} + isFinalStep={isFinalStep} + onComplete={onComplete} + onNext={() => { + setActiveStepIndex(index + 1); + }} + onToggle={() => { + setActiveStepIndex(index); + onToggleSteps?.(); + }} + /> + ); + })} +
+
+ ); }; export default GetStarted;