From c3ce92b5cb8ba47170d13b393895665268fe1d61 Mon Sep 17 00:00:00 2001 From: Adithya Vardhan Date: Mon, 7 Oct 2024 20:20:54 +0530 Subject: [PATCH] fix: white screen of death on iOS (#152) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * fix: add onboarding state to app store * fix: check if navigation state is initialized * fix: consistent naming, cleanup * fix: move redirect to _layout --------- Co-authored-by: René Aaron --- app/(app)/_layout.tsx | 27 ++++++++++++++------------- hooks/useOnboarding.ts | 18 ------------------ hooks/useSession.tsx | 4 ++-- lib/state/appStore.ts | 22 +++++++++++++++------- pages/Home.tsx | 18 +----------------- pages/Onboarding.tsx | 5 ++--- pages/settings/Settings.tsx | 4 +--- pages/settings/wallets/EditWallet.tsx | 7 ++----- 8 files changed, 37 insertions(+), 68 deletions(-) delete mode 100644 hooks/useOnboarding.ts diff --git a/app/(app)/_layout.tsx b/app/(app)/_layout.tsx index 579b2a6..dfe443c 100644 --- a/app/(app)/_layout.tsx +++ b/app/(app)/_layout.tsx @@ -1,25 +1,19 @@ import { Redirect, Stack } from 'expo-router'; import { useSession } from '~/hooks/useSession'; import { useHandleLinking } from '~/hooks/useHandleLinking'; -import { secureStorage } from '~/lib/secureStorage'; -import { hasOnboardedKey } from '~/lib/state/appStore'; -import { useMemo } from 'react'; // Add useMemo +import { useAppStore } from '~/lib/state/appStore'; +import { useRouteInfo } from 'expo-router/build/hooks'; export default function AppLayout() { const { hasSession } = useSession(); + const isOnboarded = useAppStore(store => store.isOnboarded); + const nwcClient = useAppStore(store => store.nwcClient); + const selectedWalletId = useAppStore((store) => store.selectedWalletId); + const route = useRouteInfo(); useHandleLinking(); - // Memoize the onboarded status to prevent unnecessary reads from storage - const isOnboarded = useMemo(() => { - return secureStorage.getItem(hasOnboardedKey); - }, []); - - // Don't render while the onboarding state is loaded - if (isOnboarded === null) { - return null; - } - if (!isOnboarded) { + console.log("Not onboarded, redirecting to /onboarding") return ; } @@ -28,5 +22,12 @@ export default function AppLayout() { return ; } + const connectionPage = `/settings/wallets/${selectedWalletId}/wallet-connection`; + // Check the current pathname to prevent redirect loops + if (!nwcClient && route.pathname !== connectionPage) { + console.log("No NWC client available, redirecting to wallet setup"); + return ; + } + return ; } diff --git a/hooks/useOnboarding.ts b/hooks/useOnboarding.ts deleted file mode 100644 index 3b3c781..0000000 --- a/hooks/useOnboarding.ts +++ /dev/null @@ -1,18 +0,0 @@ -import { hasOnboardedKey } from "lib/state/appStore"; -import React from "react"; -import { useEffect } from "react"; -import { secureStorage } from "~/lib/secureStorage"; - -export function useOnboarding() { - const [onboarded, setOnboarded] = React.useState(false); - useEffect(() => { - async function checkOnboardingStatus() { - const hasOnboarded = await secureStorage.getItem(hasOnboardedKey); - setOnboarded(!!hasOnboarded); - }; - - checkOnboardingStatus(); - }); - - return onboarded; -} diff --git a/hooks/useSession.tsx b/hooks/useSession.tsx index 2c6629d..0ace343 100644 --- a/hooks/useSession.tsx +++ b/hooks/useSession.tsx @@ -25,8 +25,8 @@ export function useSession() { export function SessionProvider({ children }: PropsWithChildren) { const appStore = useAppStore(); - const isSecurityEnabled = useAppStore(x => x.isSecurityEnabled); - const unlocked = useAppStore(x => x.unlocked) + const isSecurityEnabled = useAppStore(store => store.isSecurityEnabled); + const unlocked = useAppStore(store => store.unlocked) return ( void; + setOnboarded: (isOnboarded: boolean) => void; setNWCClient: (nwcClient: NWCClient | undefined) => void; setNostrWalletConnectUrl(nostrWalletConnectUrl: string): void; removeNostrWalletConnectUrl(): void; @@ -23,7 +25,6 @@ interface AppState { addWallet(wallet: Wallet): void; addAddressBookEntry(entry: AddressBookEntry): void; reset(): void; - showOnboarding(): void; getLastAlbyPayment(): Date | null; updateLastAlbyPayment(): void; } @@ -32,10 +33,10 @@ const walletKeyPrefix = "wallet"; const addressBookEntryKeyPrefix = "addressBookEntry"; const selectedWalletIdKey = "selectedWalletId"; const fiatCurrencyKey = "fiatCurrency"; +const hasOnboardedKey = "hasOnboarded"; +const lastAlbyPaymentKey = "lastAlbyPayment"; export const isSecurityEnabledKey = "isSecurityEnabled"; -export const hasOnboardedKey = "hasOnboarded"; export const lastActiveTimeKey = "lastActiveTime"; -const lastAlbyPaymentKey = "lastAlbyPayment"; type Wallet = { name?: string; @@ -116,6 +117,7 @@ export const useAppStore = create()((set, get) => { nwcClient: undefined, selectedWalletId: 0, wallets: [{}], + isOnboarded: false, }); return; } @@ -153,12 +155,21 @@ export const useAppStore = create()((set, get) => { nwcClient: getNWCClient(initialSelectedWalletId), fiatCurrency: secureStorage.getItem(fiatCurrencyKey) || "", isSecurityEnabled: iSecurityEnabled, + isOnboarded: secureStorage.getItem(hasOnboardedKey) === "true", selectedWalletId: initialSelectedWalletId, updateCurrentWallet, removeCurrentWallet, setUnlocked: (unlocked) => { set({ unlocked }); }, + setOnboarded: (isOnboarded) => { + if (isOnboarded) { + secureStorage.setItem(hasOnboardedKey, "true"); + } else { + secureStorage.removeItem(hasOnboardedKey); + } + set({ isOnboarded }); + }, setNWCClient: (nwcClient) => set({ nwcClient }), removeNostrWalletConnectUrl: () => { updateCurrentWallet({ @@ -221,10 +232,6 @@ export const useAppStore = create()((set, get) => { updateLastAlbyPayment: () => { secureStorage.setItem(lastAlbyPaymentKey, new Date().toString()); }, - showOnboarding() { - // clear onboarding status - secureStorage.removeItem(hasOnboardedKey); - }, reset() { // clear wallets for (let i = 0; i < get().wallets.length; i++) { @@ -258,6 +265,7 @@ export const useAppStore = create()((set, get) => { wallets: [{}], addressBookEntries: [], isSecurityEnabled: false, + isOnboarded: false, }); }, }; diff --git a/pages/Home.tsx b/pages/Home.tsx index 685d0bf..7cbe617 100644 --- a/pages/Home.tsx +++ b/pages/Home.tsx @@ -2,9 +2,8 @@ import { View, Pressable, StyleSheet, TouchableOpacity } from "react-native"; import React, { useState } from "react"; import { useBalance } from "hooks/useBalance"; import { useAppStore } from "lib/state/appStore"; -import { WalletConnection } from "~/pages/settings/wallets/WalletConnection"; import { - Link, router, useFocusEffect + Link, useFocusEffect } from "expo-router"; import dayjs from "dayjs"; import relativeTime from "dayjs/plugin/relativeTime"; @@ -19,7 +18,6 @@ import LargeArrowDown from "~/components/icons/LargeArrowDown"; import { SvgProps } from "react-native-svg"; import { Button } from "~/components/ui/button"; import Screen from "~/components/Screen"; -import { useOnboarding } from "~/hooks/useOnboarding"; import AlbyBanner from "~/components/AlbyBanner"; dayjs.extend(relativeTime); @@ -39,24 +37,10 @@ export function Home() { BalanceState.SATS, ); - const isOnboarded = useOnboarding(); - const hasNwcClient = !!nwcClient; - useFocusEffect(() => { reloadBalance(); }); - - React.useEffect(() => { - if (!hasNwcClient && isOnboarded) { - router.replace(`/settings/wallets/${selectedWalletId}/wallet-connection`); - } - }, [hasNwcClient, isOnboarded]); - - if (!nwcClient && isOnboarded) { - return ; - } - function switchBalanceState(): void { if (balanceState == BalanceState.SATS) { setBalanceState(BalanceState.FIAT); diff --git a/pages/Onboarding.tsx b/pages/Onboarding.tsx index 282183d..a3982eb 100644 --- a/pages/Onboarding.tsx +++ b/pages/Onboarding.tsx @@ -4,12 +4,11 @@ import React from "react"; import { View, Image } from "react-native"; import { Button } from "~/components/ui/button"; import { Text } from "~/components/ui/text"; -import { secureStorage } from "~/lib/secureStorage"; -import { hasOnboardedKey } from "~/lib/state/appStore"; +import { useAppStore } from "~/lib/state/appStore"; export function Onboarding() { async function finish() { - secureStorage.setItem(hasOnboardedKey, "true"); + useAppStore.getState().setOnboarded(true) router.replace("/"); } diff --git a/pages/settings/Settings.tsx b/pages/settings/Settings.tsx index 27218fb..1c57879 100644 --- a/pages/settings/Settings.tsx +++ b/pages/settings/Settings.tsx @@ -86,8 +86,7 @@ export function Settings() { className="flex flex-row gap-4" onPress={() => { router.dismissAll(); - useAppStore.getState().showOnboarding(); - router.replace("/onboarding"); + useAppStore.getState().setOnboarded(false); }} > @@ -109,7 +108,6 @@ export function Settings() { onPress: () => { router.dismissAll(); useAppStore.getState().reset(); - router.replace("/onboarding"); }, }, ], diff --git a/pages/settings/wallets/EditWallet.tsx b/pages/settings/wallets/EditWallet.tsx index 3aa5d56..1c72eaf 100644 --- a/pages/settings/wallets/EditWallet.tsx +++ b/pages/settings/wallets/EditWallet.tsx @@ -117,13 +117,10 @@ export function EditWallet() { { text: "Confirm", onPress: () => { - if (useAppStore.getState().wallets.length == 1) { - router.dismissAll(); - router.replace("/onboarding"); - } else { + useAppStore.getState().removeCurrentWallet(); + if (wallets.length != 1) { router.back(); } - useAppStore.getState().removeCurrentWallet(); }, }, ],