Skip to content

Commit

Permalink
login states
Browse files Browse the repository at this point in the history
  • Loading branch information
wslyvh committed Oct 31, 2024
1 parent 27c2a4e commit bac7395
Show file tree
Hide file tree
Showing 5 changed files with 137 additions and 63 deletions.
6 changes: 6 additions & 0 deletions devcon-app/next.config.js
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,12 @@ const nextConfig = {
transpilePackages: ['@parcnet-js/podspec', '@pcd/pod'],
staticPageGenerationTimeout: 300,
images: {
remotePatterns: [
{
protocol: 'https',
hostname: '**',
},
],
domains: [
'speak.devcon.org',
'storage.googleapis.com',
Expand Down
51 changes: 35 additions & 16 deletions devcon-app/src/components/domain/app/account/wallet.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ export function WalletLoginButton({ onError }: Props) {
const { address } = useAccount()
const { open } = useAppKit()
const { signMessageAsync } = useSignMessage()
const [state, setState] = useState('')
const accountContext = useAccountContext()
const router = useRouter()
const loggedIn = !!accountContext.account
Expand All @@ -29,25 +30,40 @@ export function WalletLoginButton({ onError }: Props) {
return
}

const token = await accountContext.getToken(address.toLowerCase(), false)
if (!token) {
onError?.('Unable to create verification token')
let nonce = 0
let message = ''
let signature = ''
onError?.('')

try {
setState('Sign Message')
const token = await accountContext.getToken(address.toLowerCase(), false)
if (!token) {
onError?.('Unable to create verification token')
return
}

nonce = token.nonce
message = createSiweMessage({
address: address,
chainId: 1,
domain: 'app.devcon.org',
nonce: nonce.toString(),
statement: `Sign this message to prove you have access to this wallet. This won't cost you anything.`,
uri: 'https://app.devcon.org/',
version: '1',
})
signature = await signMessageAsync({ message })
} catch (error) {
onError?.('Unable to sign message')
return
} finally {
setState('')
}

const message = createSiweMessage({
address: address,
chainId: 1,
domain: 'app.devcon.org',
nonce: token.nonce.toString(),
statement: `Sign this message to prove you have access to this wallet. This won't cost you anything.`,
uri: 'https://app.devcon.org/',
version: '1',
})

try {
const signature = await signMessageAsync({ message })
const userAccount = await accountContext.loginWeb3(address.toLowerCase(), token.nonce, message, signature)
setState('Connecting...')
const userAccount = await accountContext.loginWeb3(address.toLowerCase(), nonce, message, signature)
if (userAccount && userAccount.onboarded) {
router.push('/')
return
Expand All @@ -61,6 +77,8 @@ export function WalletLoginButton({ onError }: Props) {
}
} catch (error) {
onError?.('Unable to login')
} finally {
setState('')
}
}

Expand All @@ -85,14 +103,15 @@ export function WalletLoginButton({ onError }: Props) {
fill
className="w-full plain mt-4"
color="purple-2"
disabled={state !== ''}
onClick={(e: any) => {
e.preventDefault()
setTimeout(() => {
connectWeb3AndLogin()
}, 0)
}}
>
Continue With Ethereum
{state || 'Continue With Ethereum'}
</Button>
</>
)
Expand Down
21 changes: 11 additions & 10 deletions devcon-app/src/components/domain/app/dc7/dashboard/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@ import CalendarIcon from 'assets/icons/calendar.svg'
import { Link } from 'components/common/link'
import { TruncateMiddle } from 'utils/formatting'
import ChevronRight from 'assets/icons/chevron_right.svg'
import { FancyLoader } from 'lib/components/loader/loader'

export const cardClass =
'flex flex-col lg:border lg:border-solid lg:border-[#E4E6EB] rounded-3xl relative lg:bg-[#fbfbfb]'
Expand Down Expand Up @@ -159,13 +160,19 @@ export const Dashboard = () => {
const sessions = useRecoilValue(sessionsAtom)
const draggableLink = useDraggableLink()
const [_, setDevaBotVisible] = useRecoilState(devaBotVisibleAtom)
const loggedIn = accountContext.account
const { account, loading } = accountContext

return (
<div className={cn(cardClass, 'lg:py-4 col-start-1 col-end-4')}>
{loggedIn ? (
<LoggedIn />
) : (
{loading && !account && (
<>
<div className="flex justify-center items-center h-full">
<FancyLoader loading={loading} size={60} />
</div>
</>
)}
{!loading && account && <LoggedIn />}
{!loading && !account && (
<div className="flex justify-between md:items-center gap-6 flex-col md:flex-row px-4 relative">
<NotLoggedIn />
<div className="flex flex-col gap-2 my-2 md:w-[50%] w-full md:order-2 order-1 shrink-0">
Expand All @@ -185,7 +192,6 @@ export const Dashboard = () => {
</div>
</div>
)}

<div className="flex justify-between gap-3 pb-4 mx-4 font-semibold border-top py-4 mt-4">
Notifications
<div
Expand All @@ -195,13 +201,10 @@ export const Dashboard = () => {
<p>Go to Notifications</p>
</div>
</div>

<Notifications />

<div className="flex gap-3 pb-4 mx-4 justify-between font-semibold border-top py-4">
<div>Featured</div>
</div>

<div
className="overflow-hidden mb-6"
// style={{ maskImage: 'linear-gradient(to right, black 95%, transparent)' }}
Expand Down Expand Up @@ -288,9 +291,7 @@ export const Dashboard = () => {
</div>
</SwipeToScroll>
</div>

<div className="pb-4 mx-4 border-top"></div>

<PersonalizedSuggestions sessions={sessions || []} standalone />
</div>
)
Expand Down
21 changes: 15 additions & 6 deletions devcon-app/src/context/account-context-provider.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -47,6 +47,8 @@ export const AccountContextProvider = ({ children }: AccountContextProviderProps
await getAccount()
} catch (e) {
console.log(e, 'Account fetch failed')
} finally {
setContext({ ...context, loading: false })
}
}

Expand Down Expand Up @@ -90,7 +92,7 @@ export const AccountContextProvider = ({ children }: AccountContextProviderProps

const body = await response.json()
if (response.status === 200) {
setContext({ ...context, account: body.data })
setContext({ ...context, account: body.data, loading: false })
return body.data
}

Expand All @@ -110,7 +112,7 @@ export const AccountContextProvider = ({ children }: AccountContextProviderProps

const body = await response.json()
if (response.status === 200) {
setContext({ ...context, account: body.data })
setContext({ ...context, account: body.data, loading: false })
return body.data
}
}
Expand All @@ -127,7 +129,7 @@ export const AccountContextProvider = ({ children }: AccountContextProviderProps

const body = await response.json()
if (response.status === 200) {
setContext({ ...context, account: body.data })
setContext({ ...context, account: body.data, loading: false })
return body.data
}
}
Expand Down Expand Up @@ -180,7 +182,7 @@ export const AccountContextProvider = ({ children }: AccountContextProviderProps
if (!response) return false

if (response.status === 200) {
setContext({ ...context, account: account })
setContext({ ...context, account: account, loading: false })
return true
}

Expand All @@ -202,7 +204,7 @@ export const AccountContextProvider = ({ children }: AccountContextProviderProps

if (response.status === 200) {
const { data } = await response.json()
setContext({ ...context, account: data })
setContext({ ...context, account: data, loading: false })
return true
}

Expand Down Expand Up @@ -349,7 +351,14 @@ export const AccountContextProvider = ({ children }: AccountContextProviderProps
You need to be logged in to personalize (and share) your schedule, track your favorite speakers, and
more.
</p>
<Button color="purple-2" fill onClick={() => router.push('/login')}>
<Button
color="purple-2"
fill
onClick={() => {
setShowLoginRequired(false)
router.push('/login')
}}
>
Login
</Button>
</div>
Expand Down
101 changes: 70 additions & 31 deletions devcon-app/src/pages/login/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -103,7 +103,7 @@ const MobileLogin = (props: any) => {
type: 'spring',
bounce: 0.35,
}}
onClick={e => e.stopPropagation()}
onClick={(e: any) => e.stopPropagation()}
className="absolute bottom-0 mx-8 bg-white max-w-[500px] min-w-[300px] self-center rounded-2xl p-4 px-4 z-20 mb-8"
>
<TrustModels mobile setLoginOpen={setLoginOpen} skipLogin={props.skipLogin} />
Expand All @@ -121,6 +121,8 @@ const TrustModels = (props: any) => {
const [error, setError] = useState('')
const [email, setEmail] = useState('')
const [nonce, setNonce] = useState('')
const [loading, setLoading] = useState(false)
const [resendCooldown, setResendCooldown] = useState(0)

useEffect(() => {
async function LoginWithToken() {
Expand All @@ -140,22 +142,32 @@ const TrustModels = (props: any) => {
}, [router.query.token])

const connectEmail = async () => {
if (!isEmail(email)) {
setError('Please provide a valid email address.')
return
} else {
setError('')
}
if (loading) return

try {
setLoading(true)
if (!isEmail(email)) {
setError('Please provide a valid email address.')
return
} else {
setError('')
}

setEmailSent(true)
const token = await accountContext.getToken(email, false)
if (!token) {
setEmailSent(false)
setError('Unable to create verification token')
setEmailSent(true)
const token = await accountContext.getToken(email, false)
if (!token) {
setEmailSent(false)
setError('Unable to create verification token')
}
} finally {
setLoading(false)
}
}

const verifyEmail = async () => {
if (loading) return
setLoading(true)

const nonceNr = Number(nonce)
if (isNaN(nonceNr)) {
setError('Please provide a valid verification code.')
Expand All @@ -166,25 +178,46 @@ const TrustModels = (props: any) => {
return
}

const userAccount = await accountContext.loginEmail(email, nonceNr)
if (userAccount && userAccount.onboarded) {
router.push('/')
}
if (userAccount && !userAccount.onboarded) {
router.push('/onboarding')
}
if (!userAccount) {
try {
const userAccount = await accountContext.loginEmail(email, nonceNr)
if (userAccount && userAccount.onboarded) {
router.push('/')
}
if (userAccount && !userAccount.onboarded) {
router.push('/onboarding')
}
if (!userAccount) {
setError('Unable to verify your email address.')
}
} catch (e) {
setError('Unable to verify your email address.')
} finally {
setLoading(false)
}
}

const resendVerificationEmail = async () => {
const token = await accountContext.getToken(email, false)
if (token) {
setEmailSent(true)
} else {
setEmailSent(false)
setError('Unable to create verification token')
if (resendCooldown > 0) return

try {
setResendCooldown(30)
const token = await accountContext.getToken(email, false)
if (token) {
setEmailSent(true)
} else {
setEmailSent(false)
setError('Unable to create verification token')
}
} finally {
const interval = setInterval(() => {
setResendCooldown(prev => {
if (prev <= 1) {
clearInterval(interval)
return 0
}
return prev - 1
})
}, 1000)
}
}

Expand Down Expand Up @@ -263,11 +296,12 @@ const TrustModels = (props: any) => {
<Button
fat
fill
disabled={loading}
className="w-full plain mt-4 border !border-[#E1E4EA] border-solid"
color="grey-1"
onClick={connectEmail}
>
Continue With Email
{loading ? 'Sending Email...' : 'Continue With Email'}
</Button>
</div>

Expand Down Expand Up @@ -328,13 +362,18 @@ const TrustModels = (props: any) => {
<InputOTPSlot index={7} />
</InputOTPGroup>
</InputOTP>
<Button fat fill className="w-full plain mt-4" color="purple-2" onClick={verifyEmail}>
Verify Your Email
<Button fat fill disabled={loading} className="w-full plain mt-4" color="purple-2" onClick={verifyEmail}>
{loading ? 'Verifying Email...' : 'Verify Your Email'}
</Button>
<Separator className="mt-6 mb-4" />
<div className="flex flex-row justify-between items-center">
<div className="text-sm text-underline cursor-pointer font-semibold" onClick={resendVerificationEmail}>
Resend Verification Code
<div
className={`text-sm ${
resendCooldown > 0 ? 'text-gray-400' : 'text-underline cursor-pointer font-semibold'
}`}
onClick={resendVerificationEmail}
>
{resendCooldown > 0 ? `Resend in ${resendCooldown}s` : 'Resend Verification Code'}
</div>
<div className="text-xs cursor-pointer">Help?</div>
</div>
Expand Down

0 comments on commit bac7395

Please sign in to comment.