Skip to content

Commit

Permalink
feat: add otp auth example to expo example
Browse files Browse the repository at this point in the history
  • Loading branch information
iykazrji committed Dec 19, 2024
1 parent bccb6e7 commit 8958a88
Show file tree
Hide file tree
Showing 5 changed files with 301 additions and 128 deletions.
27 changes: 27 additions & 0 deletions examples/react-native-expo-example/app/(tabs)/_layout.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
import FontAwesome6 from "@expo/vector-icons/FontAwesome";
import { Tabs } from "expo-router";

export default function TabLayout() {
return (
<Tabs screenOptions={{ tabBarActiveTintColor: "blue" }}>
<Tabs.Screen
name="index" // Magic Link Auth
options={{
title: "Magic Link",
tabBarIcon: ({ color }) => (
<FontAwesome6 size={28} name="magic" color={color} />
),
}}
/>
<Tabs.Screen
name="otp-auth"
options={{
title: "OTP Auth",
tabBarIcon: ({ color }) => (
<FontAwesome6 size={28} name="key" color={color} />
),
}}
/>
</Tabs>
);
}
Original file line number Diff line number Diff line change
Expand Up @@ -18,11 +18,11 @@ import {
} from "react-native";
import { API_KEY } from "@env";

const signer = new RNAlchemySigner({
const signer = RNAlchemySigner({
client: { connection: { apiKey: API_KEY } },
});

export default function HomeScreen() {
export default function MagicLinkAuthScreen() {
const [email, setEmail] = useState<string>("");
const [user, setUser] = useState<User | null>(null);
const [account, setAccount] = useState<LightAccount | null>(null);
Expand Down
180 changes: 180 additions & 0 deletions examples/react-native-expo-example/app/(tabs)/otp-auth.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,180 @@
/* eslint-disable import/extensions */
import type { User } from "@account-kit/signer";
import { useEffect, useState } from "react";
import {
View,
Text,
TextInput,
StyleSheet,
TouchableOpacity,
} from "react-native";

import { API_KEY } from "@env";
import { RNAlchemySigner } from "@account-kit/react-native-signer";
import {
createLightAccountAlchemyClient,
LightAccount,
} from "@account-kit/smart-contracts";
import { sepolia } from "@account-kit/infra";
import { alchemy } from "@account-kit/infra";

const signer = RNAlchemySigner({
client: { connection: { apiKey: API_KEY! } },
});

export default function OTPAuthScreen() {
const [email, setEmail] = useState<string>("");
const [user, setUser] = useState<User | null>(null);
const [account, setAccount] = useState<LightAccount | null>(null);
const [signerAddress, setSignerAddress] = useState<string | null>(null);

const [awaitingOtp, setAwaitingOtp] = useState<boolean>(false);

const [otpCode, setOtpCode] = useState<string>("");

const handleUserAuth = ({ code }: { code: string }) => {
setAwaitingOtp(false);
signer
.authenticate({
otpCode: code,
type: "otp",
})
.then(setUser)
.catch(console.error);
};

useEffect(() => {
// get the user if already logged in
signer.getAuthDetails().then(setUser);
}, []);

useEffect(() => {
if (user) {
createLightAccountAlchemyClient({
signer,
chain: sepolia,
transport: alchemy({ apiKey: API_KEY! }),
}).then((client) => {
setAccount(client.account);
});

signer.getAddress().then((address) => {
setSignerAddress(address);
});
}
}, [user]);

return (
<View style={styles.container}>
{awaitingOtp ? (
<>
<TextInput
style={styles.textInput}
placeholderTextColor="gray"
placeholder="enter your OTP code"
onChangeText={setOtpCode}
value={otpCode}
/>
<TouchableOpacity
style={styles.button}
onPress={() => handleUserAuth({ code: otpCode })}
>
<Text style={styles.buttonText}>Sign in</Text>
</TouchableOpacity>
</>
) : !user ? (
<>
<TextInput
style={styles.textInput}
placeholderTextColor="gray"
placeholder="enter your email"
onChangeText={setEmail}
value={email}
/>
<TouchableOpacity
style={styles.button}
onPress={() => {
signer
.authenticate({
email,
type: "email",
emailMode: "otp",
})
.catch(console.error);
setAwaitingOtp(true);
}}
>
<Text style={styles.buttonText}>Sign in</Text>
</TouchableOpacity>
</>
) : (
<>
<Text style={styles.userText}>
Currently logged in as: {user.email}
</Text>
<Text style={styles.userText}>OrgId: {user.orgId}</Text>
<Text style={styles.userText}>Address: {user.address}</Text>
<Text style={styles.userText}>
Light Account Address: {account?.address}
</Text>
<Text style={styles.userText}>
Signer Address: {signerAddress}
</Text>

<TouchableOpacity
style={styles.button}
onPress={() =>
signer.disconnect().then(() => setUser(null))
}
>
<Text style={styles.buttonText}>Sign out</Text>
</TouchableOpacity>
</>
)}
</View>
);
}

const styles = StyleSheet.create({
container: {
flex: 1,
alignItems: "center",
justifyContent: "center",
backgroundColor: "#FFFFF",
paddingHorizontal: 20,
},
textInput: {
width: "100%",
height: 40,
borderColor: "gray",
borderWidth: 1,
paddingHorizontal: 10,
backgroundColor: "rgba(0,0,0,0.05)",
marginTop: 20,
marginBottom: 10,
},
box: {
width: 60,
height: 60,
marginVertical: 20,
},
button: {
width: 200,
padding: 10,
height: 50,
backgroundColor: "rgb(147, 197, 253)",
borderRadius: 5,
alignItems: "center",
justifyContent: "center",
marginTop: 20,
},
buttonText: {
color: "white",
fontWeight: "bold",
textAlign: "center",
},
userText: {
marginBottom: 10,
fontSize: 18,
},
});
8 changes: 6 additions & 2 deletions examples/react-native-expo-example/app/_layout.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -3,8 +3,12 @@ import "node-libs-react-native/globals.js";
import "react-native-get-random-values";

import React from "react";
import { Slot } from "expo-router";
import { Stack } from "expo-router";

export default function RootLayout() {
return <Slot />;
return (
<Stack>
<Stack.Screen name="(tabs)" options={{ headerShown: false }} />
</Stack>
);
}
Loading

0 comments on commit 8958a88

Please sign in to comment.