mirror of
https://github.com/logto-io/logto.git
synced 2025-01-13 21:30:30 -05:00
refactor(console): update swift guide (#6123)
This commit is contained in:
parent
9904ac74d9
commit
07e3725740
10 changed files with 116 additions and 54 deletions
|
@ -1,6 +1,3 @@
|
||||||
import InlineNotification from '@/ds-components/InlineNotification';
|
|
||||||
import UriInputField from '@/mdx-components/UriInputField';
|
|
||||||
|
|
||||||
Before we dive into the details, here's a quick overview of the end-user experience. The sign-in process can be simplified as follows:
|
Before we dive into the details, here's a quick overview of the end-user experience. The sign-in process can be simplified as follows:
|
||||||
|
|
||||||
```mermaid
|
```mermaid
|
||||||
|
@ -21,15 +18,3 @@ Regarding redirect-based sign-in:
|
||||||
To learn more about the rationale and benefits of redirect-based sign-in, see [Logto sign-in experience explained](https://docs.logto.io/docs/tutorials/get-started/sign-in-experience).
|
To learn more about the rationale and benefits of redirect-based sign-in, see [Logto sign-in experience explained](https://docs.logto.io/docs/tutorials/get-started/sign-in-experience).
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
<InlineNotification>
|
|
||||||
In the following steps, we assume your app is running on <code>http://localhost:3000</code>.
|
|
||||||
</InlineNotification>
|
|
||||||
|
|
||||||
Now, let's enter your redirect URI. E.g. {`${props.callbackUri ?? 'http://localhost:3000/callback'}`}.
|
|
||||||
|
|
||||||
<UriInputField name="redirectUris" />
|
|
||||||
|
|
||||||
Just like signing in, users should be redirected to Logto for signing out of the shared session. Once finished, it would be great to redirect the user back to your website. For example, add `http://localhost:3000` as the post sign-out redirect URI below.
|
|
||||||
|
|
||||||
<UriInputField name="postLogoutRedirectUris" />
|
|
|
@ -0,0 +1,10 @@
|
||||||
|
import InlineNotification from '@/ds-components/InlineNotification';
|
||||||
|
import UriInputField from '@/mdx-components/UriInputField';
|
||||||
|
|
||||||
|
import ExperienceOverview from './_experience-overview.md';
|
||||||
|
|
||||||
|
<ExperienceOverview />
|
||||||
|
|
||||||
|
Now, let's configure your redirect URI. E.g. {`${props.defaultUri ?? 'http://localhost:3000/callback'}`}.
|
||||||
|
|
||||||
|
<UriInputField name="redirectUris" />
|
|
@ -0,0 +1,18 @@
|
||||||
|
import InlineNotification from '@/ds-components/InlineNotification';
|
||||||
|
import UriInputField from '@/mdx-components/UriInputField';
|
||||||
|
|
||||||
|
import ExperienceOverview from './_experience-overview.md';
|
||||||
|
|
||||||
|
<ExperienceOverview />
|
||||||
|
|
||||||
|
<InlineNotification>
|
||||||
|
In the following steps, we assume your app is running on <code>http://localhost:3000</code>.
|
||||||
|
</InlineNotification>
|
||||||
|
|
||||||
|
Now, let's configure your redirect URI. E.g. {`${props.defaultUri ?? 'http://localhost:3000/callback'}`}.
|
||||||
|
|
||||||
|
<UriInputField name="redirectUris" />
|
||||||
|
|
||||||
|
Just like signing in, users should be redirected to Logto for signing out of the shared session. Once finished, it would be great to redirect the user back to your website. For example, add `http://localhost:3000` as the post sign-out redirect URI below.
|
||||||
|
|
||||||
|
<UriInputField name="postLogoutRedirectUris" />
|
|
@ -4,6 +4,8 @@ import TabItem from '@mdx/components/TabItem';
|
||||||
import InlineNotification from '@/ds-components/InlineNotification';
|
import InlineNotification from '@/ds-components/InlineNotification';
|
||||||
import Steps from '@/mdx-components/Steps';
|
import Steps from '@/mdx-components/Steps';
|
||||||
import Step from '@/mdx-components/Step';
|
import Step from '@/mdx-components/Step';
|
||||||
|
import Checkpoint from '../../fragments/_checkpoint.md';
|
||||||
|
import RedirectUrisNative from '../../fragments/_redirect-uris-native.mdx';
|
||||||
|
|
||||||
<Steps>
|
<Steps>
|
||||||
|
|
||||||
|
@ -44,6 +46,8 @@ CocoaPods [does not support local dependency](https://github.com/CocoaPods/Cocoa
|
||||||
subtitle="1 step"
|
subtitle="1 step"
|
||||||
>
|
>
|
||||||
|
|
||||||
|
You can initialize `LogtoClient` in a proper place of your app that can be accessed globally:
|
||||||
|
|
||||||
<Code className="language-swift">
|
<Code className="language-swift">
|
||||||
{`import Logto
|
{`import Logto
|
||||||
import LogtoClient
|
import LogtoClient
|
||||||
|
@ -68,50 +72,93 @@ let config = try? LogtoConfig(
|
||||||
|
|
||||||
</Step>
|
</Step>
|
||||||
|
|
||||||
<Step
|
<Step title="Configure redirect URI">
|
||||||
title="Sign in"
|
|
||||||
subtitle="2 steps"
|
|
||||||
>
|
|
||||||
|
|
||||||
### Configure Redirect URI
|
<RedirectUrisNative />
|
||||||
|
|
||||||
First, let’s configure your redirect URI scheme. E.g. `io.logto://callback`
|
|
||||||
|
|
||||||
<UriInputField name="redirectUris" />
|
|
||||||
|
|
||||||
<InlineNotification>
|
<InlineNotification>
|
||||||
The Redirect URI in iOS SDK is only for internal use. There's <em>NO NEED</em> to add a{' '}
|
The Redirect URI in iOS SDK is only for internal use. There's <em>NO NEED</em> to add a [Custom URL Scheme](https://developer.apple.com/documentation/xcode/defining-a-custom-url-scheme-for-your-app) until a connector asks.
|
||||||
<a href="https://developer.apple.com/documentation/xcode/defining-a-custom-url-scheme-for-your-app">
|
|
||||||
Custom URL Scheme
|
|
||||||
</a>{' '}
|
|
||||||
until a connector asks.
|
|
||||||
</InlineNotification>
|
</InlineNotification>
|
||||||
|
|
||||||
Go back to Xcode, use the following code to implement sign-in:
|
</Step>
|
||||||
|
|
||||||
<Code className="language-swift">
|
<Step title="Implement sign-in and sign-out">
|
||||||
{`do {
|
|
||||||
|
You can use `client.signInWithBrowser(redirectUri:)` to sign in the user and `client.signOut()` to sign out the user.
|
||||||
|
|
||||||
|
For example, in a SwiftUI app:
|
||||||
|
|
||||||
|
<Code title="ContentView.swift" className="language-swift">
|
||||||
|
{`struct ContentView: View {
|
||||||
|
@State var isAuthenticated: Bool
|
||||||
|
|
||||||
|
init() {
|
||||||
|
isAuthenticated = client.isAuthenticated
|
||||||
|
}
|
||||||
|
|
||||||
|
var body: some View {
|
||||||
|
VStack {
|
||||||
|
if isAuthenticated {
|
||||||
|
Button("Sign Out") {
|
||||||
|
Task { [self] in
|
||||||
|
await client.signOut()
|
||||||
|
isAuthenticated = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
Button("Sign In") {
|
||||||
|
Task { [self] in
|
||||||
|
do {
|
||||||
try await client.signInWithBrowser(redirectUri: "${
|
try await client.signInWithBrowser(redirectUri: "${
|
||||||
props.redirectUris[0] ?? 'io.logto://callback'
|
props.redirectUris[0] ?? 'io.logto://callback'
|
||||||
}")
|
}")
|
||||||
print(client.isAuthenticated) // true
|
isAuthenticated = true
|
||||||
} catch let error as LogtoClientErrors.SignIn {
|
} catch let error as LogtoClientErrors.SignIn {
|
||||||
// error occured during sign in
|
// error occured during sign in
|
||||||
|
} catch {
|
||||||
|
// other errors
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}`}
|
}`}
|
||||||
</Code>
|
</Code>
|
||||||
|
|
||||||
</Step>
|
</Step>
|
||||||
|
|
||||||
<Step
|
<Step title="Checkpoint: Test your app">
|
||||||
title="Sign out"
|
|
||||||
subtitle="1 step"
|
|
||||||
>
|
|
||||||
|
|
||||||
Calling `.signOut()` will clean all the Logto data in Keychain, if they exist.
|
<Checkpoint />
|
||||||
|
|
||||||
```swift
|
</Step>
|
||||||
await client.signOut()
|
|
||||||
```
|
<Step title="Display user information">
|
||||||
|
|
||||||
|
To display the user's information, you can use the `client.getIdTokenClaims()` method. For example, in a SwiftUI app:
|
||||||
|
|
||||||
|
<Code title="ContentView.swift" className="language-swift">
|
||||||
|
{`struct ContentView: View {
|
||||||
|
@State var isAuthenticated: Bool
|
||||||
|
@State var name: String?
|
||||||
|
|
||||||
|
init() {
|
||||||
|
isAuthenticated = client.isAuthenticated
|
||||||
|
name = try? client.getIdTokenClaims().name
|
||||||
|
}
|
||||||
|
|
||||||
|
var body: some View {
|
||||||
|
VStack {
|
||||||
|
if isAuthenticated {
|
||||||
|
Text("Welcome, \(name)")
|
||||||
|
} else {
|
||||||
|
Text("Please sign in")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}`}
|
||||||
|
</Code>
|
||||||
|
|
||||||
</Step>
|
</Step>
|
||||||
|
|
||||||
|
|
|
@ -10,6 +10,7 @@ const metadata: Readonly<GuideMetadata> = Object.freeze({
|
||||||
repo: 'swift',
|
repo: 'swift',
|
||||||
path: 'Demos/SwiftUI%20Demo',
|
path: 'Demos/SwiftUI%20Demo',
|
||||||
},
|
},
|
||||||
|
fullGuide: 'swift',
|
||||||
});
|
});
|
||||||
|
|
||||||
export default metadata;
|
export default metadata;
|
||||||
|
|
|
@ -4,7 +4,7 @@ import { generateStandardSecret } from '@logto/shared/universal';
|
||||||
import Steps from '@/mdx-components/Steps';
|
import Steps from '@/mdx-components/Steps';
|
||||||
import Step from '@/mdx-components/Step';
|
import Step from '@/mdx-components/Step';
|
||||||
import Checkpoint from '../../fragments/_checkpoint.md';
|
import Checkpoint from '../../fragments/_checkpoint.md';
|
||||||
import RedirectUris from '../../fragments/_redirect_uris.mdx';
|
import RedirectUrisWeb from '../../fragments/_redirect-uris-web.mdx';
|
||||||
|
|
||||||
<Steps>
|
<Steps>
|
||||||
|
|
||||||
|
@ -84,7 +84,7 @@ export async function GET(request: NextRequest) {
|
||||||
subtitle="2 URIs"
|
subtitle="2 URIs"
|
||||||
>
|
>
|
||||||
|
|
||||||
<RedirectUris />
|
<RedirectUrisWeb />
|
||||||
|
|
||||||
</Step>
|
</Step>
|
||||||
|
|
||||||
|
|
|
@ -4,7 +4,7 @@ import { generateStandardSecret } from '@logto/shared/universal';
|
||||||
import Steps from '@/mdx-components/Steps';
|
import Steps from '@/mdx-components/Steps';
|
||||||
import Step from '@/mdx-components/Step';
|
import Step from '@/mdx-components/Step';
|
||||||
import Checkpoint from '../../fragments/_checkpoint.md';
|
import Checkpoint from '../../fragments/_checkpoint.md';
|
||||||
import RedirectUris from '../../fragments/_redirect_uris.mdx';
|
import RedirectUrisWeb from '../../fragments/_redirect-uris-web.mdx';
|
||||||
|
|
||||||
<Steps>
|
<Steps>
|
||||||
|
|
||||||
|
@ -85,7 +85,7 @@ This will create 4 routes automatically:
|
||||||
subtitle="2 URIs"
|
subtitle="2 URIs"
|
||||||
>
|
>
|
||||||
|
|
||||||
<RedirectUris callbackUri="http://localhost:3000/api/logto/sign-in-callback" />
|
<RedirectUrisWeb defaultUri="http://localhost:3000/api/logto/sign-in-callback" />
|
||||||
|
|
||||||
</Step>
|
</Step>
|
||||||
|
|
||||||
|
|
|
@ -5,7 +5,7 @@ import InlineNotification from '@/ds-components/InlineNotification';
|
||||||
import Steps from '@/mdx-components/Steps';
|
import Steps from '@/mdx-components/Steps';
|
||||||
import Step from '@/mdx-components/Step';
|
import Step from '@/mdx-components/Step';
|
||||||
import Checkpoint from '../../fragments/_checkpoint.md';
|
import Checkpoint from '../../fragments/_checkpoint.md';
|
||||||
import RedirectUris from '../../fragments/_redirect_uris.mdx';
|
import RedirectUrisWeb from '../../fragments/_redirect-uris-web.mdx';
|
||||||
import { generateStandardSecret } from '@logto/shared/universal';
|
import { generateStandardSecret } from '@logto/shared/universal';
|
||||||
|
|
||||||
export const cookieEncryptionKey = generateStandardSecret();
|
export const cookieEncryptionKey = generateStandardSecret();
|
||||||
|
@ -84,7 +84,7 @@ See [runtime config](https://nuxt.com/docs/guide/going-further/runtime-config) f
|
||||||
subtitle="2 URIs"
|
subtitle="2 URIs"
|
||||||
>
|
>
|
||||||
|
|
||||||
<RedirectUris />
|
<RedirectUrisWeb />
|
||||||
|
|
||||||
When registering `@logto/nuxt` module, it will do the following:
|
When registering `@logto/nuxt` module, it will do the following:
|
||||||
|
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
import UriInputField from '@/mdx-components/UriInputField';
|
import UriInputField from '@/mdx-components/UriInputField';
|
||||||
import InlineNotification from '@/ds-components/InlineNotification';
|
import InlineNotification from '@/ds-components/InlineNotification';
|
||||||
import { generateStandardSecret } from '@logto/shared/universal';
|
import { generateStandardSecret } from '@logto/shared/universal';
|
||||||
import RedirectUris from '../../fragments/_redirect_uris.mdx';
|
import RedirectUrisWeb from '../../fragments/_redirect-uris-web.mdx';
|
||||||
import Checkpoint from '../../fragments/_checkpoint.md';
|
import Checkpoint from '../../fragments/_checkpoint.md';
|
||||||
import Steps from '@/mdx-components/Steps';
|
import Steps from '@/mdx-components/Steps';
|
||||||
import Step from '@/mdx-components/Step';
|
import Step from '@/mdx-components/Step';
|
||||||
|
@ -82,7 +82,7 @@ end`}
|
||||||
subtitle="2 URIs"
|
subtitle="2 URIs"
|
||||||
>
|
>
|
||||||
|
|
||||||
<RedirectUris />
|
<RedirectUrisWeb />
|
||||||
|
|
||||||
</Step>
|
</Step>
|
||||||
|
|
||||||
|
|
|
@ -12,6 +12,7 @@ const metadata: Readonly<GuideMetadata> = Object.freeze({
|
||||||
repo: 'ruby',
|
repo: 'ruby',
|
||||||
path: 'logto-sample',
|
path: 'logto-sample',
|
||||||
},
|
},
|
||||||
|
fullGuide: 'ruby',
|
||||||
});
|
});
|
||||||
|
|
||||||
export default metadata;
|
export default metadata;
|
||||||
|
|
Loading…
Add table
Reference in a new issue