From 4931923e1c9f58c0da0eebad49f11cfb9c45d30a Mon Sep 17 00:00:00 2001 From: Charles Zhao Date: Tue, 24 May 2022 16:22:43 +0800 Subject: [PATCH] feat(console): add integration guide for vue sdk --- .../docs/tutorial/integrate-sdk/vue.mdx | 196 ++++++++++++++++++ .../docs/tutorial/integrate-sdk/vue_zh-cn.mdx | 196 ++++++++++++++++++ .../console/src/assets/images/congrats.svg | 33 +++ .../components/GuideModal/index.module.scss | 2 +- .../components/GuideModal/index.tsx | 15 +- .../LibrarySelector/index.module.scss | 5 + .../components/LibrarySelector/index.tsx | 4 +- .../StepsSkeleton/index.module.scss | 40 ++++ .../components/StepsSkeleton/index.tsx | 20 ++ 9 files changed, 503 insertions(+), 8 deletions(-) create mode 100644 packages/console/src/assets/docs/tutorial/integrate-sdk/vue.mdx create mode 100644 packages/console/src/assets/docs/tutorial/integrate-sdk/vue_zh-cn.mdx create mode 100644 packages/console/src/assets/images/congrats.svg create mode 100644 packages/console/src/pages/Applications/components/StepsSkeleton/index.module.scss create mode 100644 packages/console/src/pages/Applications/components/StepsSkeleton/index.tsx diff --git a/packages/console/src/assets/docs/tutorial/integrate-sdk/vue.mdx b/packages/console/src/assets/docs/tutorial/integrate-sdk/vue.mdx new file mode 100644 index 000000000..867a43bb9 --- /dev/null +++ b/packages/console/src/assets/docs/tutorial/integrate-sdk/vue.mdx @@ -0,0 +1,196 @@ +import MultiTextInputField from '@mdx/components/MultiTextInputField'; +import Step from '@mdx/components/Step'; +import Tabs from '@mdx/components/Tabs'; +import TabItem from '@mdx/components/TabItem'; + + props.onNext(1)} +> + + + +```bash +npm i @logto/vue +``` + + + + +```bash +yarn add @logto/vue +``` + + + + +```bash +pnpm add @logto/vue +``` + + + + +{/* This should be CDN URL */} +```html + + + +``` + +Next, we need to link the callback component with the route. Let's say the path is `/callback` and we are using `vue-router`: + +```ts +const router = createRouter({ + routes: [ + { + path: "/callback", + name: "callback", + component: CallbackView, + }, + ] +}); +``` + +### Make a Sign In Button + +```ts +import { useLogto } from "@logto/vue"; + +const { signIn } = useLogto(); +const onClickSignIn = () => signIn(redirectUrl); +``` + +```html + +``` + +### Retrieve Authentication Status + +```ts +import { useLogto } from '@logto/vue'; + +const { isAuthenticated } = useLogto(); +``` + +```html +
+ +
+
+ +
+``` + +
+ + props.onNext(4)} +> + +Calling `.signOut()` will clear all the Logto data in memory and LocalStorage, if there is any. + +To make the user come back to your application after signing out, +it's necessary to add `http://localhost:1234` as one of the Post Sign Out URIs and use the URL as the parameter when calling `.signOut()`. + + props.onError(3)} /> + +```ts +import { useLogto } from "@logto/vue"; + +const { signOut } = useLogto(); +const onClickSignOut = () => signOut('http://localhost:1234'); +``` + +```html + +``` + + + + + +- [SDK Documentation](https://link-url-here.org) +- [OIDC Documentation](https://link-url-here.org) +- [Calling API to fetch accessToken](https://link-url-here.org) + + diff --git a/packages/console/src/assets/docs/tutorial/integrate-sdk/vue_zh-cn.mdx b/packages/console/src/assets/docs/tutorial/integrate-sdk/vue_zh-cn.mdx new file mode 100644 index 000000000..cbd66f2bc --- /dev/null +++ b/packages/console/src/assets/docs/tutorial/integrate-sdk/vue_zh-cn.mdx @@ -0,0 +1,196 @@ +import MultiTextInputField from '@mdx/components/MultiTextInputField'; +import Step from '@mdx/components/Step'; +import Tabs from '@mdx/components/Tabs'; +import TabItem from '@mdx/components/TabItem'; + + props.onNext(1)} +> + + + +```bash +npm i @logto/vue +``` + + + + +```bash +yarn add @logto/vue +``` + + + + +```bash +pnpm add @logto/vue +``` + + + + +{/* This should be CDN URL */} +```html + + + +``` + +接下来,我们就可以在路由表中添加这个回调路由。假设我们的路由地址定义为 `/callback`,且使用的路由工具为 `vue-router`: + +```ts +const router = createRouter({ + ... + routes: [ + { + path: "/callback", + name: "callback", + component: CallbackView, + }, + ] +}); +``` + +### 创建一个登录按钮 + +```ts +import { useLogto } from "@logto/vue"; + +const { signIn } = useLogto(); +const onClickSignIn = () => signIn(redirectUrl); +``` + +```html + +``` + +### 判断当前登录状态 + +```ts +import { useLogto } from '@logto/vue'; + +const { isAuthenticated } = useLogto(); +``` + +```html +
+ +
+
+ +
+``` + +
+ + props.onNext(4)} +> + +调用 `.signOut()` 方法会清除所有在缓存或者 localStorage 中的 Logto 数据(如果有)。 + +为了确保用户登出后能够跳转回你的应用,我们需要首先在管理界面中将 `http://localhost:1234` 添加到允许登出后跳转的地址列表(Post Sign Out URIs)中。 + + props.onError(3)} /> + +```ts +import { useLogto } from "@logto/vue"; + +const { signOut } = useLogto(); +const onClickSignOut = () => signOut('http://localhost:1234'); +``` + +```html + +``` + + + + + +- [SDK Documentation](https://link-url-here.org) +- [OIDC Documentation](https://link-url-here.org) +- [Calling API to fetch accessToken](https://link-url-here.org) + + diff --git a/packages/console/src/assets/images/congrats.svg b/packages/console/src/assets/images/congrats.svg new file mode 100644 index 000000000..2f282cda9 --- /dev/null +++ b/packages/console/src/assets/images/congrats.svg @@ -0,0 +1,33 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/packages/console/src/pages/Applications/components/GuideModal/index.module.scss b/packages/console/src/pages/Applications/components/GuideModal/index.module.scss index 423330558..35edeefba 100644 --- a/packages/console/src/pages/Applications/components/GuideModal/index.module.scss +++ b/packages/console/src/pages/Applications/components/GuideModal/index.module.scss @@ -3,7 +3,7 @@ .container { display: flex; flex-direction: column; - background-color: var(--color-surface-1); + background-color: var(--color-base); height: 100vh; .header { diff --git a/packages/console/src/pages/Applications/components/GuideModal/index.tsx b/packages/console/src/pages/Applications/components/GuideModal/index.tsx index 0114e7dcb..ea1a06ddb 100644 --- a/packages/console/src/pages/Applications/components/GuideModal/index.tsx +++ b/packages/console/src/pages/Applications/components/GuideModal/index.tsx @@ -17,6 +17,7 @@ import { SupportedJavascriptLibraries } from '@/types/applications'; import { GuideForm } from '@/types/guide'; import LibrarySelector from '../LibrarySelector'; +import StepsSkeleton from '../StepsSkeleton'; import * as styles from './index.module.scss'; type Props = { @@ -28,7 +29,9 @@ type Props = { const Guides: Record JSX.Element>> = { react: lazy(async () => import('@/assets/docs/tutorial/integrate-sdk/react.mdx')), + vue: lazy(async () => import('@/assets/docs/tutorial/integrate-sdk/vue.mdx')), 'react_zh-cn': lazy(async () => import('@/assets/docs/tutorial/integrate-sdk/react_zh-cn.mdx')), + 'vue_zh-cn': lazy(async () => import('@/assets/docs/tutorial/integrate-sdk/vue_zh-cn.mdx')), }; const onClickFetchSampleProject = (projectName: string) => { @@ -37,13 +40,15 @@ const onClickFetchSampleProject = (projectName: string) => { }; const GuideModal = ({ appName, isOpen, onClose, onComplete }: Props) => { - const [subtype, setSubtype] = useState(SupportedJavascriptLibraries.React); + const [subtype, setSubtype] = useState( + SupportedJavascriptLibraries.React + ); const [activeStepIndex, setActiveStepIndex] = useState(-1); const [invalidStepIndex, setInvalidStepIndex] = useState(-1); const locale = i18next.language; - const guideKey = `${subtype}_${locale}`.toLowerCase(); - const GuideComponent = Guides[guideKey] ?? Guides[subtype]; + const guideI18nKey = `${subtype}_${locale}`.toLowerCase(); + const GuideComponent = Guides[guideI18nKey] ?? Guides[subtype]; const methods = useForm({ mode: 'onSubmit', reValidateMode: 'onChange' }); const { @@ -84,7 +89,7 @@ const GuideModal = ({ appName, isOpen, onClose, onComplete }: Props) => {
- {cloneElement(, { + {cloneElement(, { className: styles.banner, onChange: setSubtype, onToggle: () => { @@ -100,7 +105,7 @@ const GuideModal = ({ appName, isOpen, onClose, onComplete }: Props) => { }, }} > - Loading...
}> + }> {GuideComponent && ( ( - success + success
{t('applications.guide.title')}
{t('applications.guide.subtitle')}
diff --git a/packages/console/src/pages/Applications/components/StepsSkeleton/index.module.scss b/packages/console/src/pages/Applications/components/StepsSkeleton/index.module.scss new file mode 100644 index 000000000..ca2252314 --- /dev/null +++ b/packages/console/src/pages/Applications/components/StepsSkeleton/index.module.scss @@ -0,0 +1,40 @@ +@use '@/scss/underscore' as _; + +.step { + display: flex; + align-items: center; + padding: _.unit(5) _.unit(6); + border-radius: 16px; + background-color: var(--color-layer-1); + + .index { + @include _.shimmering-animation; + width: 28px; + height: 28px; + border-radius: 50%; + margin-right: _.unit(4); + } + + .wrapper { + flex: 1; + display: flex; + flex-direction: column; + + .title { + @include _.shimmering-animation; + width: 140px; + height: 24px; + } + + .subtitle { + @include _.shimmering-animation; + width: 400px; + height: 20px; + margin-top: _.unit(1); + } + } +} + +.step + .step { + margin-top: _.unit(6); +} diff --git a/packages/console/src/pages/Applications/components/StepsSkeleton/index.tsx b/packages/console/src/pages/Applications/components/StepsSkeleton/index.tsx new file mode 100644 index 000000000..accb090df --- /dev/null +++ b/packages/console/src/pages/Applications/components/StepsSkeleton/index.tsx @@ -0,0 +1,20 @@ +import React from 'react'; + +import * as styles from './index.module.scss'; + +const StepsSkeleton = () => ( + <> + {Array.from({ length: 5 }).map((_, stepIndex) => ( + // eslint-disable-next-line react/no-array-index-key +
+
+
+
+
+
+
+ ))} + +); + +export default StepsSkeleton;