diff --git a/.changeset/spicy-flowers-return.md b/.changeset/spicy-flowers-return.md new file mode 100644 index 00000000000..af1fe47f8b5 --- /dev/null +++ b/.changeset/spicy-flowers-return.md @@ -0,0 +1,12 @@ +--- +"thirdweb": minor +--- + +Adds SIWF for in-app wallets + +```ts +await wallet.connect({ + strategy: "farcaster", + client: CLIENT, +}); +``` diff --git a/apps/playground-web/src/lib/constants.ts b/apps/playground-web/src/lib/constants.ts index ae5f9ba541b..f94bc7fd442 100644 --- a/apps/playground-web/src/lib/constants.ts +++ b/apps/playground-web/src/lib/constants.ts @@ -9,12 +9,12 @@ export const WALLETS = [ auth: { options: [ "google", - "facebook", "discord", "apple", "email", "passkey", "phone", + "farcaster" ], mode: "redirect", }, diff --git a/packages/thirdweb/src/react/core/utils/socialIcons.ts b/packages/thirdweb/src/react/core/utils/socialIcons.ts index 63a2e0e17e9..432a0392fc6 100644 --- a/packages/thirdweb/src/react/core/utils/socialIcons.ts +++ b/packages/thirdweb/src/react/core/utils/socialIcons.ts @@ -19,6 +19,8 @@ export const twitchIconUri = ""; export const discordIconUri = ""; +export const farcasterIconUri = + ""; export const microsoftIconUri = ""; @@ -48,4 +50,5 @@ export const socialIcons = { apple: appleIconUri, facebook: facebookIconUri, discord: discordIconUri, + farcaster: farcasterIconUri, }; diff --git a/packages/thirdweb/src/react/native/ui/components/WalletImage.tsx b/packages/thirdweb/src/react/native/ui/components/WalletImage.tsx index fc67b411be0..b8913f64ae9 100644 --- a/packages/thirdweb/src/react/native/ui/components/WalletImage.tsx +++ b/packages/thirdweb/src/react/native/ui/components/WalletImage.tsx @@ -10,6 +10,7 @@ import { DISCORD_ICON, EMAIL_ICON, FACEBOOK_ICON, + FARCASTER_ICON, GOOGLE_ICON, PHONE_ICON, WALLET_ICON, @@ -76,6 +77,8 @@ export function getAuthProviderImage(lastAuthProvider: string | null): string { return FACEBOOK_ICON; case "discord": return DISCORD_ICON; + case "farcaster": + return FARCASTER_ICON; default: return WALLET_ICON; } diff --git a/packages/thirdweb/src/react/native/ui/connect/InAppWalletUI.tsx b/packages/thirdweb/src/react/native/ui/connect/InAppWalletUI.tsx index 0a935a6f99f..c062a35a3ea 100644 --- a/packages/thirdweb/src/react/native/ui/connect/InAppWalletUI.tsx +++ b/packages/thirdweb/src/react/native/ui/connect/InAppWalletUI.tsx @@ -13,6 +13,10 @@ import type { } from "../../../../wallets/in-app/core/wallet/types.js"; import { preAuthenticate } from "../../../../wallets/in-app/native/auth/index.js"; import type { Wallet } from "../../../../wallets/interfaces/wallet.js"; +import { + type SocialAuthOption, + socialAuthOptions, +} from "../../../../wallets/types.js"; import type { Theme } from "../../../core/design-system/index.js"; import { setLastAuthProvider } from "../../../core/utils/storage.js"; import { radius, spacing } from "../../design-system/index.js"; @@ -26,6 +30,7 @@ import { DISCORD_ICON, EMAIL_ICON, FACEBOOK_ICON, + FARCASTER_ICON, GOOGLE_ICON, PHONE_ICON, } from "../icons/svgs.js"; @@ -44,6 +49,7 @@ const socialIcons = { facebook: FACEBOOK_ICON, apple: APPLE_ICON, discord: DISCORD_ICON, + farcaster: FARCASTER_ICON, }; type InAppWalletFormUIProps = { @@ -62,9 +68,8 @@ export function InAppWalletUI(props: InAppWalletFormUIProps) { const { wallet, theme } = props; const config = wallet.getConfig(); const authOptions = config?.auth?.options || defaultAuthOptions; - const socialLogins = authOptions.filter( - (x) => - x === "google" || x === "apple" || x === "facebook" || x === "discord", + const socialLogins = authOptions.filter((x) => + socialAuthOptions.includes(x as SocialAuthOption), ) as InAppWalletSocialAuth[]; const [inputMode, setInputMode] = useState<"email" | "phone">("email"); diff --git a/packages/thirdweb/src/react/native/ui/icons/svgs.ts b/packages/thirdweb/src/react/native/ui/icons/svgs.ts index 0725ffc3758..d655eac47de 100644 --- a/packages/thirdweb/src/react/native/ui/icons/svgs.ts +++ b/packages/thirdweb/src/react/native/ui/icons/svgs.ts @@ -37,6 +37,13 @@ export const DISCORD_ICON = ` `; +export const FARCASTER_ICON = ` + + + + +`; + export const EMAIL_ICON = ` diff --git a/packages/thirdweb/src/react/web/ui/components/WalletImage.tsx b/packages/thirdweb/src/react/web/ui/components/WalletImage.tsx index a5d49a24dd2..476208ea4e2 100644 --- a/packages/thirdweb/src/react/web/ui/components/WalletImage.tsx +++ b/packages/thirdweb/src/react/web/ui/components/WalletImage.tsx @@ -12,6 +12,7 @@ import { discordIconUri, emailIcon, facebookIconUri, + farcasterIconUri, genericWalletIcon, googleIconUri, passkeyIcon, @@ -77,6 +78,9 @@ export function WalletImage(props: { case "discord": image = discordIconUri; break; + case "farcaster": + image = farcasterIconUri; + break; } } else { const mipdImage = getInstalledWalletProviders().find( diff --git a/packages/thirdweb/src/react/web/wallets/shared/ConnectWalletSocialOptions.tsx b/packages/thirdweb/src/react/web/wallets/shared/ConnectWalletSocialOptions.tsx index 82900f562c8..1b2b745038e 100644 --- a/packages/thirdweb/src/react/web/wallets/shared/ConnectWalletSocialOptions.tsx +++ b/packages/thirdweb/src/react/web/wallets/shared/ConnectWalletSocialOptions.tsx @@ -91,6 +91,7 @@ export const ConnectWalletSocialOptions = ( facebook: locale.signInWithFacebook, apple: locale.signInWithApple, discord: locale.signInWithDiscord, + farcaster: "Farcaster", }; const { data: ecosystemAuthOptions, isLoading } = useQuery({ diff --git a/packages/thirdweb/src/react/web/wallets/shared/oauthSignIn.ts b/packages/thirdweb/src/react/web/wallets/shared/oauthSignIn.ts index 5c9fd4b3d6f..ab9dee30225 100644 --- a/packages/thirdweb/src/react/web/wallets/shared/oauthSignIn.ts +++ b/packages/thirdweb/src/react/web/wallets/shared/oauthSignIn.ts @@ -33,6 +33,7 @@ function getOauthLoginPath( case "apple": case "facebook": case "google": + case "farcaster": case "discord": return getSocialAuthLoginPath(authOption, client, ecosystem); default: diff --git a/packages/thirdweb/src/wallets/in-app/core/authentication/type.ts b/packages/thirdweb/src/wallets/in-app/core/authentication/type.ts index 916f900ac56..cd01604cfba 100644 --- a/packages/thirdweb/src/wallets/in-app/core/authentication/type.ts +++ b/packages/thirdweb/src/wallets/in-app/core/authentication/type.ts @@ -62,6 +62,7 @@ export enum AuthProvider { APPLE = "Apple", PASSKEY = "Passkey", DISCORD = "Discord", + FARCASTER = "Farcaster", } export type OauthOption = { diff --git a/packages/thirdweb/src/wallets/in-app/native/native-connector.ts b/packages/thirdweb/src/wallets/in-app/native/native-connector.ts index 90d22afb1ee..cf5eefde379 100644 --- a/packages/thirdweb/src/wallets/in-app/native/native-connector.ts +++ b/packages/thirdweb/src/wallets/in-app/native/native-connector.ts @@ -109,6 +109,7 @@ export class InAppNativeConnector implements InAppConnector { case "google": case "facebook": case "discord": + case "farcaster": case "apple": { const ExpoLinking = require("expo-linking"); const redirectUrl = diff --git a/packages/thirdweb/src/wallets/in-app/web/lib/auth/oauth.ts b/packages/thirdweb/src/wallets/in-app/web/lib/auth/oauth.ts index ee8e6b91c45..d273069208a 100644 --- a/packages/thirdweb/src/wallets/in-app/web/lib/auth/oauth.ts +++ b/packages/thirdweb/src/wallets/in-app/web/lib/auth/oauth.ts @@ -62,12 +62,13 @@ export const loginWithOauth = async (options: { let win = options.openedWindow; let isWindowOpenedByFn = false; if (!win) { + const redirectUrl = new URL(window.location.href); win = window.open( - getSocialAuthLoginPath( + `${getSocialAuthLoginPath( options.authOption, options.client, options.ecosystem, - ), + )}&redirectUrl=${encodeURIComponent(redirectUrl.toString())}`, `Login to ${options.authOption}`, DEFAULT_POP_UP_SIZE, ); diff --git a/packages/thirdweb/src/wallets/in-app/web/lib/web-connector.ts b/packages/thirdweb/src/wallets/in-app/web/lib/web-connector.ts index 99f18eea8d1..159d5ba56a3 100644 --- a/packages/thirdweb/src/wallets/in-app/web/lib/web-connector.ts +++ b/packages/thirdweb/src/wallets/in-app/web/lib/web-connector.ts @@ -222,6 +222,7 @@ export class InAppWebConnector implements InAppConnector { case "apple": case "facebook": case "google": + case "farcaster": case "discord": { const authToken = await loginWithOauth({ authOption: strategy, diff --git a/packages/thirdweb/src/wallets/types.ts b/packages/thirdweb/src/wallets/types.ts index 6ede9631e6f..a422b3246f4 100644 --- a/packages/thirdweb/src/wallets/types.ts +++ b/packages/thirdweb/src/wallets/types.ts @@ -24,6 +24,7 @@ export const socialAuthOptions = [ "apple", "facebook", "discord", + "farcaster", ] as const; export type SocialAuthOption = (typeof socialAuthOptions)[number];