From 9ce451b6f7d6108bf21418bae8384feedbbb7aaf Mon Sep 17 00:00:00 2001 From: Stephen Heaps Date: Tue, 28 Nov 2023 11:33:03 -0500 Subject: [PATCH 1/6] Check for wallet existing before creating Wallet to prevent possible race condition with multi-window. --- .../Crypto/Stores/KeyringStore.swift | 23 ++++++++++++++----- 1 file changed, 17 insertions(+), 6 deletions(-) diff --git a/Sources/BraveWallet/Crypto/Stores/KeyringStore.swift b/Sources/BraveWallet/Crypto/Stores/KeyringStore.swift index e903ff45438..e5972b11914 100644 --- a/Sources/BraveWallet/Crypto/Stores/KeyringStore.swift +++ b/Sources/BraveWallet/Crypto/Stores/KeyringStore.swift @@ -383,13 +383,24 @@ public class KeyringStore: ObservableObject, WalletObserverStore { } isOnboarding = true isCreatingWallet = true - keyringService.createWallet(password) { [weak self] mnemonic in - self?.isCreatingWallet = false - self?.updateInfo() - if !mnemonic.isEmpty { - self?.passwordToSaveInBiometric = password + keyringService.isWalletCreated { [weak self] isWalletCreated in + guard let self else { return } + guard !isWalletCreated else { + // Wallet was created already (possible with multi-window) #8425 + self.isOnboarding = false + self.isCreatingWallet = false + // Dismiss onboarding if wallet is already setup + self.isOnboardingVisible = false + return + } + keyringService.createWallet(password) { mnemonic in + self.isCreatingWallet = false + self.updateInfo() + if !mnemonic.isEmpty { + self.passwordToSaveInBiometric = password + } + completion?(mnemonic) } - completion?(mnemonic) } } From adcca34e754a5a45437d5ea9c96d988feabc1163 Mon Sep 17 00:00:00 2001 From: Stephen Heaps Date: Tue, 28 Nov 2023 14:27:26 -0500 Subject: [PATCH 2/6] Add 'Creating Wallet...' step to onboarding for when asynchronous create wallet / restore wallet functions are setting up the wallet and fetching wallet data files. --- Sources/BraveWallet/Crypto/CryptoView.swift | 4 +- .../Crypto/Onboarding/CreateWalletView.swift | 97 ++++++++++++++----- .../Crypto/Onboarding/LegalView.swift | 26 +++-- .../Crypto/Onboarding/RestoreWalletView.swift | 74 ++++++++++---- .../Crypto/Onboarding/SetupCryptoView.swift | 20 +++- .../BraveWallet/Crypto/UnlockWalletView.swift | 8 +- Sources/BraveWallet/WalletStrings.swift | 7 ++ 7 files changed, 176 insertions(+), 60 deletions(-) diff --git a/Sources/BraveWallet/Crypto/CryptoView.swift b/Sources/BraveWallet/Crypto/CryptoView.swift index 0b3bef368ff..d62dcbb8eb2 100644 --- a/Sources/BraveWallet/Crypto/CryptoView.swift +++ b/Sources/BraveWallet/Crypto/CryptoView.swift @@ -236,7 +236,7 @@ public struct CryptoView: View { } case .unlock: UIKitNavigationView { - UnlockWalletView(keyringStore: keyringStore) + UnlockWalletView(keyringStore: keyringStore, dismissAction: dismissAction) .toolbar { dismissButtonToolbarContents } @@ -252,7 +252,7 @@ public struct CryptoView: View { .zIndex(2) // Needed or the dismiss animation messes up } else { UIKitNavigationView { - SetupCryptoView(keyringStore: keyringStore) + SetupCryptoView(keyringStore: keyringStore, dismissAction: dismissAction) .toolbar { ToolbarItemGroup(placement: .destructiveAction) { Button(action: { diff --git a/Sources/BraveWallet/Crypto/Onboarding/CreateWalletView.swift b/Sources/BraveWallet/Crypto/Onboarding/CreateWalletView.swift index 88c4fc48f16..b028f80e7b4 100644 --- a/Sources/BraveWallet/Crypto/Onboarding/CreateWalletView.swift +++ b/Sources/BraveWallet/Crypto/Onboarding/CreateWalletView.swift @@ -5,32 +5,37 @@ import Foundation import SwiftUI +import BraveUI import DesignSystem import Strings import struct Shared.AppConstants -struct RestorePackage { - let recoveryWords: [String] - let onRestoreCompleted: (_ status: Bool, _ validPassword: String) -> Void - var isLegacyWallet: Bool { - recoveryWords.count == .legacyWalletRecoveryPhraseNumber - } -} - struct CreateWalletContainerView: View { @ObservedObject var keyringStore: KeyringStore - var restorePackage: RestorePackage? + let setupOption: OnboardingSetupOption + let onValidPasswordEntered: ((_ validPassword: String) -> Void)? + // Used to dismiss all of Wallet + let dismissAction: () -> Void - init(keyringStore: KeyringStore, restorePackage: RestorePackage? = nil) { + init( + keyringStore: KeyringStore, + setupOption: OnboardingSetupOption, + onValidPasswordEntered: ((_ validPassword: String) -> Void)? = nil, + dismissAction: @escaping () -> Void + ) { self.keyringStore = keyringStore - self.restorePackage = restorePackage + self.setupOption = setupOption + self.onValidPasswordEntered = onValidPasswordEntered + self.dismissAction = dismissAction } var body: some View { ScrollView(.vertical) { CreateWalletView( keyringStore: keyringStore, - restorePackage: restorePackage + setupOption: setupOption, + onValidPasswordEntered: onValidPasswordEntered, + dismissAction: dismissAction ) .background(Color(.braveBackground)) } @@ -55,7 +60,10 @@ private enum ValidationError: LocalizedError, Equatable { private struct CreateWalletView: View { @ObservedObject var keyringStore: KeyringStore - var restorePackage: RestorePackage? + let setupOption: OnboardingSetupOption + let onValidPasswordEntered: ((_ validPassword: String) -> Void)? + // Used to dismiss all of Wallet + let dismissAction: () -> Void @State private var password: String = "" @State private var repeatedPassword: String = "" @@ -63,25 +71,27 @@ private struct CreateWalletView: View { @State private var isNewWalletCreated: Bool = false @State private var passwordStatus: PasswordStatus = .none @State private var isInputsMatch: Bool = false + /// If this view is showing `Creating Wallet...` overlay, blocking input fields. + /// Using a local flag for the view instead of `keyringStore.isCreatingWallet` so we + /// only show `CreatingWalletView` on the `RestoreWalletView` when restoring. + @State private var isShowingCreatingWallet: Bool = false @FocusState private var isFieldFocused: Bool private func createWallet() { - if let restorePackage { - // restore wallet with recovery phrases and a new password - keyringStore.restoreWallet( - words: restorePackage.recoveryWords, - password: password, - isLegacyBraveWallet: restorePackage.isLegacyWallet - ) { success in - restorePackage.onRestoreCompleted(success, password) - } - } else { + switch setupOption { + case .new: + isShowingCreatingWallet = true keyringStore.createWallet(password: password) { mnemonic in + defer { self.isShowingCreatingWallet = false } if let mnemonic, !mnemonic.isEmpty { isNewWalletCreated = true } } + case .restore: + if isInputsMatch { + onValidPasswordEntered?(password) + } } } @@ -224,6 +234,23 @@ private struct CreateWalletView: View { ) .onChange(of: password, perform: handleInputChange) .onChange(of: repeatedPassword, perform: handleInputChange) + .navigationBarBackButtonHidden(isShowingCreatingWallet) + .overlay { + if isShowingCreatingWallet { + CreatingWalletView() + } + } + .toolbar(content: { + ToolbarItem(placement: .topBarLeading) { + if isShowingCreatingWallet { + Button(action: dismissAction) { // dismiss all of wallet + Image("wallet-dismiss", bundle: .module) + .renderingMode(.template) + .foregroundColor(Color(.braveBlurpleTint)) + } + } + } + }) .onAppear { isFieldFocused = true } @@ -234,10 +261,32 @@ private struct CreateWalletView: View { struct CreateWalletView_Previews: PreviewProvider { static var previews: some View { NavigationView { - CreateWalletContainerView(keyringStore: .previewStore) + CreateWalletContainerView( + keyringStore: .previewStore, + setupOption: .new, + dismissAction: {} + ) } .previewLayout(.sizeThatFits) .previewColorSchemes() } } #endif + +/// View shown as an overlay over `CreateWalletView` or `RestoreWalletView` +/// when waiting for Wallet to be created & wallet data files downloaded. +struct CreatingWalletView: View { + + var body: some View { + VStack(spacing: 24) { + Spacer() + ProgressView() + .progressViewStyle(.braveCircular(size: .normal, tint: .braveBlurpleTint)) + Text(Strings.Wallet.creatingWallet) + .font(.title) + Spacer() + } + .frame(maxWidth: .infinity, maxHeight: .infinity) + .background(Color(braveSystemName: .containerBackground)) + } +} diff --git a/Sources/BraveWallet/Crypto/Onboarding/LegalView.swift b/Sources/BraveWallet/Crypto/Onboarding/LegalView.swift index 8b3daffc8a3..1e192423230 100644 --- a/Sources/BraveWallet/Crypto/Onboarding/LegalView.swift +++ b/Sources/BraveWallet/Crypto/Onboarding/LegalView.swift @@ -8,18 +8,15 @@ import DesignSystem struct LegalView: View { @ObservedObject var keyringStore: KeyringStore - var setupOption: SetupOption + let setupOption: OnboardingSetupOption + // Used to dismiss all of Wallet + let dismissAction: () -> Void @State private var isResponsibilityCheckboxChecked: Bool = false @State private var isTermsCheckboxChecked: Bool = false @State private var isShowingCreateNewWallet: Bool = false @State private var isShowingRestoreExistedWallet: Bool = false - enum SetupOption { - case new - case restore - } - private var isContinueDisabled: Bool { !isResponsibilityCheckboxChecked || !isTermsCheckboxChecked } @@ -79,7 +76,11 @@ struct LegalView: View { .padding() .background( NavigationLink( - destination: CreateWalletContainerView(keyringStore: keyringStore), + destination: CreateWalletContainerView( + keyringStore: keyringStore, + setupOption: setupOption, + dismissAction: dismissAction + ), isActive: $isShowingCreateNewWallet, label: { EmptyView() @@ -88,7 +89,10 @@ struct LegalView: View { ) .background( NavigationLink( - destination: RestoreWalletContainerView(keyringStore: keyringStore), + destination: RestoreWalletContainerView( + keyringStore: keyringStore, + dismissAction: dismissAction + ), isActive: $isShowingRestoreExistedWallet, label: { EmptyView() @@ -118,7 +122,11 @@ struct LegalCheckbox: View { #if DEBUG struct LegalView_Previews: PreviewProvider { static var previews: some View { - LegalView(keyringStore: .previewStore, setupOption: .new) + LegalView( + keyringStore: .previewStore, + setupOption: .new, + dismissAction: {} + ) } } #endif diff --git a/Sources/BraveWallet/Crypto/Onboarding/RestoreWalletView.swift b/Sources/BraveWallet/Crypto/Onboarding/RestoreWalletView.swift index 23450f99f78..f06194097d3 100644 --- a/Sources/BraveWallet/Crypto/Onboarding/RestoreWalletView.swift +++ b/Sources/BraveWallet/Crypto/Onboarding/RestoreWalletView.swift @@ -12,10 +12,12 @@ import Preferences struct RestoreWalletContainerView: View { @ObservedObject var keyringStore: KeyringStore + // Used to dismiss all of Wallet + let dismissAction: () -> Void var body: some View { ScrollView(.vertical) { - RestoreWalletView(keyringStore: keyringStore) + RestoreWalletView(keyringStore: keyringStore, dismissAction: dismissAction) .background(Color(.braveBackground)) } .background(Color(.braveBackground).edgesIgnoringSafeArea(.all)) @@ -25,9 +27,10 @@ struct RestoreWalletContainerView: View { private struct RestoreWalletView: View { @ObservedObject var keyringStore: KeyringStore + // Used to dismiss all of Wallet + let dismissAction: () -> Void @Environment(\.sizeCategory) private var sizeCategory - @Environment(\.dismiss) private var dismiss @State private var isBraveLegacyWallet: Bool = false @State private var isRevealRecoveryWords: Bool = true @@ -49,6 +52,10 @@ private struct RestoreWalletView: View { !recoveryWords.allSatisfy({ !$0.isEmpty }) || keyringStore.isRestoringWallet } + private var isShowingCreatingWallet: Bool { + keyringStore.isCreatingWallet || keyringStore.isRestoringWallet + } + private var errorLabel: some View { HStack(spacing: 12) { Image(braveSystemName: "leo.warning.circle-filled") @@ -96,7 +103,6 @@ private struct RestoreWalletView: View { resignFirstResponder() } } - } var body: some View { @@ -188,26 +194,30 @@ private struct RestoreWalletView: View { .onChange(of: recoveryWords) { [recoveryWords] newValue in handleRecoveryWordsChanged(oldValue: recoveryWords, newValue: newValue) } + .navigationBarBackButtonHidden(isShowingCreatingWallet) + .overlay { + if isShowingCreatingWallet { + CreatingWalletView() + } + } + .toolbar(content: { + ToolbarItem(placement: .topBarLeading) { + if isShowingCreatingWallet { + Button(action: dismissAction) { // dismiss all of wallet + Image("wallet-dismiss", bundle: .module) + .renderingMode(.template) + .foregroundColor(Color(.braveBlurpleTint)) + } + } + } + }) .sheet(isPresented: $isShowingCreateNewPassword) { NavigationView { CreateWalletContainerView( keyringStore: keyringStore, - restorePackage: RestorePackage( - recoveryWords: recoveryWords, - onRestoreCompleted: { success, password in - if success { - isShowingPhraseError = false - keyringStore.resetKeychainStoredPassword() - if keyringStore.isOnboardingVisible { - Preferences.Wallet.isOnboardingCompleted.value = true - } - } else { - newPassword = password - isShowingPhraseError = true - } - isShowingCreateNewPassword = false - } - ) + setupOption: .restore, + onValidPasswordEntered: restoreWallet, + dismissAction: dismissAction ) .toolbar { ToolbarItemGroup(placement: .destructiveAction) { @@ -223,13 +233,37 @@ private struct RestoreWalletView: View { private func resignFirstResponder() { UIApplication.shared.sendAction(#selector(UIResponder.resignFirstResponder), to: nil, from: nil, for: nil) } + + private func restoreWallet(_ password: String) { + newPassword = password + isShowingCreateNewPassword = false + keyringStore.restoreWallet( + words: recoveryWords, + password: password, + isLegacyBraveWallet: recoveryWords.count == .legacyWalletRecoveryPhraseNumber + ) { success in + if success { + isShowingPhraseError = false + keyringStore.resetKeychainStoredPassword() + if keyringStore.isOnboardingVisible { + Preferences.Wallet.isOnboardingCompleted.value = true + } + } else { + newPassword = password + isShowingPhraseError = true + } + } + } } #if DEBUG struct RestoreWalletView_Previews: PreviewProvider { static var previews: some View { NavigationView { - RestoreWalletContainerView(keyringStore: .previewStore) + RestoreWalletContainerView( + keyringStore: .previewStore, + dismissAction: {} + ) } .previewLayout(.sizeThatFits) .previewColorSchemes() diff --git a/Sources/BraveWallet/Crypto/Onboarding/SetupCryptoView.swift b/Sources/BraveWallet/Crypto/Onboarding/SetupCryptoView.swift index b3857690919..9bd94772620 100644 --- a/Sources/BraveWallet/Crypto/Onboarding/SetupCryptoView.swift +++ b/Sources/BraveWallet/Crypto/Onboarding/SetupCryptoView.swift @@ -9,10 +9,17 @@ import Introspect import DesignSystem import Strings +enum OnboardingSetupOption { + case new + case restore +} + struct SetupCryptoView: View { @ObservedObject var keyringStore: KeyringStore + // Used to dismiss all of Wallet + let dismissAction: () -> Void - @State private var setupOption: LegalView.SetupOption? + @State private var setupOption: OnboardingSetupOption? var body: some View { ScrollView { @@ -117,7 +124,11 @@ struct SetupCryptoView: View { ), destination: { if let option = setupOption { - LegalView(keyringStore: keyringStore, setupOption: option) + LegalView( + keyringStore: keyringStore, + setupOption: option, + dismissAction: dismissAction + ) } }, label: { @@ -134,7 +145,10 @@ struct SetupCryptoView: View { struct SetupCryptoView_Previews: PreviewProvider { static var previews: some View { NavigationView { - SetupCryptoView(keyringStore: .previewStore) + SetupCryptoView( + keyringStore: .previewStore, + dismissAction: {} + ) } .previewLayout(.sizeThatFits) .previewColorSchemes() diff --git a/Sources/BraveWallet/Crypto/UnlockWalletView.swift b/Sources/BraveWallet/Crypto/UnlockWalletView.swift index 80cc2ad6bf8..977021eaf81 100644 --- a/Sources/BraveWallet/Crypto/UnlockWalletView.swift +++ b/Sources/BraveWallet/Crypto/UnlockWalletView.swift @@ -10,6 +10,8 @@ import LocalAuthentication struct UnlockWalletView: View { @ObservedObject var keyringStore: KeyringStore + // Used to dismiss all of Wallet + let dismissAction: () -> Void @State private var password: String = "" @FocusState private var isPasswordFieldFocused: Bool @@ -81,7 +83,8 @@ struct UnlockWalletView: View { NavigationLink( destination: RestoreWalletContainerView( - keyringStore: keyringStore + keyringStore: keyringStore, + dismissAction: dismissAction ) ) { Text(Strings.Wallet.restoreWalletButtonTitle) @@ -161,7 +164,8 @@ struct UnlockWalletView_Previews: PreviewProvider { static var previews: some View { NavigationView { UnlockWalletView( - keyringStore: .previewStoreWithWalletCreated + keyringStore: .previewStoreWithWalletCreated, + dismissAction: {} ) } .previewColorSchemes() diff --git a/Sources/BraveWallet/WalletStrings.swift b/Sources/BraveWallet/WalletStrings.swift index 0443f161c28..9f970965ee9 100644 --- a/Sources/BraveWallet/WalletStrings.swift +++ b/Sources/BraveWallet/WalletStrings.swift @@ -856,6 +856,13 @@ extension Strings { value: "Strong", comment: "A label will be displayed beside input password when it is considered as a strong password" ) + public static let creatingWallet = NSLocalizedString( + "wallet.creatingWallet", + tableName: "BraveWallet", + bundle: .module, + value: "Creating Wallet...", + comment: "The title of the creating wallet screen, shown after user enters their password while the wallet is being set up." + ) public static let biometricsSetupErrorTitle = NSLocalizedString( "wallet.biometricsSetupErrorTitle", tableName: "BraveWallet", From dc64045b514dc65e407b9bb367322f043359f34e Mon Sep 17 00:00:00 2001 From: Stephen Heaps Date: Thu, 30 Nov 2023 14:26:10 -0500 Subject: [PATCH 3/6] Fix Wallet re-opened before create/restore wallet callback returns. --- .../BraveWallet/Crypto/Stores/KeyringStore.swift | 16 ++++++++++++++++ 1 file changed, 16 insertions(+) diff --git a/Sources/BraveWallet/Crypto/Stores/KeyringStore.swift b/Sources/BraveWallet/Crypto/Stores/KeyringStore.swift index e5972b11914..c9a23ad1f61 100644 --- a/Sources/BraveWallet/Crypto/Stores/KeyringStore.swift +++ b/Sources/BraveWallet/Crypto/Stores/KeyringStore.swift @@ -213,6 +213,14 @@ public class KeyringStore: ObservableObject, WalletObserverStore { public func setupObservers() { guard !isObserving else { return } + Task { @MainActor in + // For case where Wallet is dismissed while wallet is being created. + // Ex. User creates wallet, dismisses & re-opens Wallet before + // create wallet completion callback. Callback is held with + // strong ref which keeps `KeyringStore` alive. + let isWalletCreated = await keyringService.isWalletCreated() + self.isOnboardingVisible = !isWalletCreated + } self.keyringServiceObserver = KeyringServiceObserver( keyringService: keyringService, _walletReset: { [weak self] in @@ -270,6 +278,14 @@ public class KeyringStore: ObservableObject, WalletObserverStore { } public func tearDown() { + Task { @MainActor in + // For case where Wallet is dismissed while wallet is being created. + // Ex. User creates wallet, dismisses & re-opens Wallet before + // create wallet completion callback. Callback is held with + // strong ref which keeps `KeyringStore` alive. + let isWalletCreated = await keyringService.isWalletCreated() + self.isOnboardingVisible = !isWalletCreated + } keyringServiceObserver = nil rpcServiceObserver = nil } From 8024465d279e2c68f64ef70964f17b768147cc70 Mon Sep 17 00:00:00 2001 From: Stephen Heaps Date: Thu, 30 Nov 2023 16:01:41 -0500 Subject: [PATCH 4/6] Use old api for `ToolbarItemPlacement` as CI is using older Xcode version. --- Sources/BraveWallet/Crypto/Onboarding/CreateWalletView.swift | 2 +- Sources/BraveWallet/Crypto/Onboarding/RestoreWalletView.swift | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/Sources/BraveWallet/Crypto/Onboarding/CreateWalletView.swift b/Sources/BraveWallet/Crypto/Onboarding/CreateWalletView.swift index b028f80e7b4..7e69e1b8578 100644 --- a/Sources/BraveWallet/Crypto/Onboarding/CreateWalletView.swift +++ b/Sources/BraveWallet/Crypto/Onboarding/CreateWalletView.swift @@ -241,7 +241,7 @@ private struct CreateWalletView: View { } } .toolbar(content: { - ToolbarItem(placement: .topBarLeading) { + ToolbarItem(placement: .navigationBarLeading) { if isShowingCreatingWallet { Button(action: dismissAction) { // dismiss all of wallet Image("wallet-dismiss", bundle: .module) diff --git a/Sources/BraveWallet/Crypto/Onboarding/RestoreWalletView.swift b/Sources/BraveWallet/Crypto/Onboarding/RestoreWalletView.swift index f06194097d3..5ce77d65d19 100644 --- a/Sources/BraveWallet/Crypto/Onboarding/RestoreWalletView.swift +++ b/Sources/BraveWallet/Crypto/Onboarding/RestoreWalletView.swift @@ -201,7 +201,7 @@ private struct RestoreWalletView: View { } } .toolbar(content: { - ToolbarItem(placement: .topBarLeading) { + ToolbarItem(placement: .navigationBarLeading) { if isShowingCreatingWallet { Button(action: dismissAction) { // dismiss all of wallet Image("wallet-dismiss", bundle: .module) From de5cd5c221eb23439167d415058f0aa6650ab8c4 Mon Sep 17 00:00:00 2001 From: Stephen Heaps Date: Mon, 4 Dec 2023 14:19:32 -0500 Subject: [PATCH 5/6] Address PR comments; Fix backup banner showing until network call to fetch wallet data returns after wallet restore. Banner would only be visible if wallet is dismissed and re-opened prior to callback. Fix wallet dismiss on wallet lock. --- Sources/BraveWallet/Crypto/Stores/KeyringStore.swift | 5 +++++ Sources/BraveWallet/WalletHostingViewController.swift | 2 +- 2 files changed, 6 insertions(+), 1 deletion(-) diff --git a/Sources/BraveWallet/Crypto/Stores/KeyringStore.swift b/Sources/BraveWallet/Crypto/Stores/KeyringStore.swift index c9a23ad1f61..20c8296f529 100644 --- a/Sources/BraveWallet/Crypto/Stores/KeyringStore.swift +++ b/Sources/BraveWallet/Crypto/Stores/KeyringStore.swift @@ -285,6 +285,11 @@ public class KeyringStore: ObservableObject, WalletObserverStore { // strong ref which keeps `KeyringStore` alive. let isWalletCreated = await keyringService.isWalletCreated() self.isOnboardingVisible = !isWalletCreated + if isRestoringWallet && isWalletCreated { + // user dismissed wallet while restoring, but after wallet was created in core. + keyringService.notifyWalletBackupComplete() + self.isWalletBackedUp = await keyringService.isWalletBackedUp() + } } keyringServiceObserver = nil rpcServiceObserver = nil diff --git a/Sources/BraveWallet/WalletHostingViewController.swift b/Sources/BraveWallet/WalletHostingViewController.swift index f6c03d4dc47..557d88fcf8d 100644 --- a/Sources/BraveWallet/WalletHostingViewController.swift +++ b/Sources/BraveWallet/WalletHostingViewController.swift @@ -95,7 +95,7 @@ public class WalletHostingViewController: UIHostingController { // As a workaround to this issue, we can just watch keyring's `isLocked` value from here // and dismiss the first sheet ourselves to ensure we dont get stuck with a child view visible // while the wallet is locked. - if /*#unavailable(iOS 16.4),*/ + if #unavailable(iOS 16.4), let self = self, isLocked, let presentedViewController = self.presentedViewController, From a244b488c75d6ea754d61208ac35001a76de410e Mon Sep 17 00:00:00 2001 From: Stephen Heaps Date: Wed, 6 Dec 2023 12:57:05 -0500 Subject: [PATCH 6/6] Address PR comment; Fix `CreatingWalletView` not covering fullscreen by overlaying on ScrollView instead of only on contents. --- .../Crypto/Onboarding/CreateWalletView.swift | 163 ++++++++---------- .../Crypto/Onboarding/LegalView.swift | 4 +- .../Crypto/Onboarding/RestoreWalletView.swift | 26 +-- .../BraveWallet/Crypto/UnlockWalletView.swift | 2 +- 4 files changed, 86 insertions(+), 109 deletions(-) diff --git a/Sources/BraveWallet/Crypto/Onboarding/CreateWalletView.swift b/Sources/BraveWallet/Crypto/Onboarding/CreateWalletView.swift index 7e69e1b8578..83997973870 100644 --- a/Sources/BraveWallet/Crypto/Onboarding/CreateWalletView.swift +++ b/Sources/BraveWallet/Crypto/Onboarding/CreateWalletView.swift @@ -10,40 +10,6 @@ import DesignSystem import Strings import struct Shared.AppConstants -struct CreateWalletContainerView: View { - @ObservedObject var keyringStore: KeyringStore - let setupOption: OnboardingSetupOption - let onValidPasswordEntered: ((_ validPassword: String) -> Void)? - // Used to dismiss all of Wallet - let dismissAction: () -> Void - - init( - keyringStore: KeyringStore, - setupOption: OnboardingSetupOption, - onValidPasswordEntered: ((_ validPassword: String) -> Void)? = nil, - dismissAction: @escaping () -> Void - ) { - self.keyringStore = keyringStore - self.setupOption = setupOption - self.onValidPasswordEntered = onValidPasswordEntered - self.dismissAction = dismissAction - } - - var body: some View { - ScrollView(.vertical) { - CreateWalletView( - keyringStore: keyringStore, - setupOption: setupOption, - onValidPasswordEntered: onValidPasswordEntered, - dismissAction: dismissAction - ) - .background(Color(.braveBackground)) - } - .background(Color(.braveBackground).edgesIgnoringSafeArea(.all)) - .transparentNavigationBar(backButtonTitle: Strings.Wallet.createWalletBackButtonTitle, backButtonDisplayMode: .generic) - } -} - private enum ValidationError: LocalizedError, Equatable { case requirementsNotMet case inputsDontMatch @@ -58,7 +24,7 @@ private enum ValidationError: LocalizedError, Equatable { } } -private struct CreateWalletView: View { +struct CreateWalletView: View { @ObservedObject var keyringStore: KeyringStore let setupOption: OnboardingSetupOption let onValidPasswordEntered: ((_ validPassword: String) -> Void)? @@ -77,6 +43,18 @@ private struct CreateWalletView: View { @State private var isShowingCreatingWallet: Bool = false @FocusState private var isFieldFocused: Bool + + init( + keyringStore: KeyringStore, + setupOption: OnboardingSetupOption, + onValidPasswordEntered: ((_ validPassword: String) -> Void)? = nil, + dismissAction: @escaping () -> Void + ) { + self.keyringStore = keyringStore + self.setupOption = setupOption + self.onValidPasswordEntered = onValidPasswordEntered + self.dismissAction = dismissAction + } private func createWallet() { switch setupOption { @@ -137,7 +115,7 @@ private struct CreateWalletView: View { } } - func errorLabel(_ error: ValidationError?) -> some View { + private func errorLabel(_ error: ValidationError?) -> some View { HStack(spacing: 12) { Image(braveSystemName: "leo.warning.circle-filled") .renderingMode(.template) @@ -161,65 +139,67 @@ private struct CreateWalletView: View { } var body: some View { - VStack(spacing: 16) { - VStack { - Text(Strings.Wallet.createWalletTitle) - .font(.title) - .padding(.bottom) - .multilineTextAlignment(.center) - .foregroundColor(Color(uiColor: WalletV2Design.textPrimary)) - Text(Strings.Wallet.createWalletSubTitle) - .font(.subheadline) - .padding(.bottom) - .multilineTextAlignment(.center) - .foregroundColor(Color(uiColor: WalletV2Design.textSecondary)) - } - VStack(alignment: .leading, spacing: 20) { - VStack(spacing: 30) { - VStack(alignment: .leading, spacing: 10) { - Text(Strings.Wallet.newPasswordPlaceholder) - .foregroundColor(Color(uiColor: WalletV2Design.textPrimary)) - HStack(spacing: 8) { - SecureField(Strings.Wallet.newPasswordPlaceholder, text: $password) - .textContentType(.newPassword) - .focused($isFieldFocused) - Spacer() - if passwordStatus != .none { - passwordStatusView(passwordStatus) + ScrollView(.vertical) { + VStack(spacing: 16) { + VStack { + Text(Strings.Wallet.createWalletTitle) + .font(.title) + .padding(.bottom) + .multilineTextAlignment(.center) + .foregroundColor(Color(uiColor: WalletV2Design.textPrimary)) + Text(Strings.Wallet.createWalletSubTitle) + .font(.subheadline) + .padding(.bottom) + .multilineTextAlignment(.center) + .foregroundColor(Color(uiColor: WalletV2Design.textSecondary)) + } + VStack(alignment: .leading, spacing: 20) { + VStack(spacing: 30) { + VStack(alignment: .leading, spacing: 10) { + Text(Strings.Wallet.newPasswordPlaceholder) + .foregroundColor(Color(uiColor: WalletV2Design.textPrimary)) + HStack(spacing: 8) { + SecureField(Strings.Wallet.newPasswordPlaceholder, text: $password) + .textContentType(.newPassword) + .focused($isFieldFocused) + Spacer() + if passwordStatus != .none { + passwordStatusView(passwordStatus) + } } + Divider() } - Divider() - } - VStack(alignment: .leading, spacing: 12) { - Text(Strings.Wallet.repeatedPasswordPlaceholder) - .foregroundColor(.primary) - HStack(spacing: 8) { - SecureField(Strings.Wallet.repeatedPasswordPlaceholder, text: $repeatedPassword, onCommit: createWallet) - .textContentType(.newPassword) - Spacer() - if isInputsMatch { - Text("\(Image(braveSystemName: "leo.check.normal")) \(Strings.Wallet.repeatedPasswordMatch)") - .multilineTextAlignment(.trailing) - .font(.footnote) - .foregroundColor(.secondary) + VStack(alignment: .leading, spacing: 12) { + Text(Strings.Wallet.repeatedPasswordPlaceholder) + .foregroundColor(.primary) + HStack(spacing: 8) { + SecureField(Strings.Wallet.repeatedPasswordPlaceholder, text: $repeatedPassword, onCommit: createWallet) + .textContentType(.newPassword) + Spacer() + if isInputsMatch { + Text("\(Image(braveSystemName: "leo.check.normal")) \(Strings.Wallet.repeatedPasswordMatch)") + .multilineTextAlignment(.trailing) + .font(.footnote) + .foregroundColor(.secondary) + } } + Divider() } - Divider() } + .font(.subheadline) + errorLabel(validationError) } - .font(.subheadline) - errorLabel(validationError) - } - Button(action: createWallet) { - Text(Strings.Wallet.continueButtonTitle) - .frame(maxWidth: .infinity) + Button(action: createWallet) { + Text(Strings.Wallet.continueButtonTitle) + .frame(maxWidth: .infinity) + } + .buttonStyle(BraveFilledButtonStyle(size: .large)) + .disabled(isContinueDisabled) + .padding(.top, 60) } - .buttonStyle(BraveFilledButtonStyle(size: .large)) - .disabled(isContinueDisabled) - .padding(.top, 60) + .padding(.horizontal, 20) + .padding(.bottom, 20) } - .padding(.horizontal, 20) - .padding(.bottom, 20) .background(Color(.braveBackground).edgesIgnoringSafeArea(.all)) .background( NavigationLink( @@ -235,9 +215,12 @@ private struct CreateWalletView: View { .onChange(of: password, perform: handleInputChange) .onChange(of: repeatedPassword, perform: handleInputChange) .navigationBarBackButtonHidden(isShowingCreatingWallet) + .frame(maxWidth: .infinity, maxHeight: .infinity) .overlay { if isShowingCreatingWallet { CreatingWalletView() + .ignoresSafeArea() + .frame(maxWidth: .infinity, maxHeight: .infinity) } } .toolbar(content: { @@ -254,6 +237,10 @@ private struct CreateWalletView: View { .onAppear { isFieldFocused = true } + .transparentNavigationBar( + backButtonTitle: Strings.Wallet.createWalletBackButtonTitle, + backButtonDisplayMode: .generic + ) } } @@ -261,7 +248,7 @@ private struct CreateWalletView: View { struct CreateWalletView_Previews: PreviewProvider { static var previews: some View { NavigationView { - CreateWalletContainerView( + CreateWalletView( keyringStore: .previewStore, setupOption: .new, dismissAction: {} diff --git a/Sources/BraveWallet/Crypto/Onboarding/LegalView.swift b/Sources/BraveWallet/Crypto/Onboarding/LegalView.swift index 1e192423230..a7c22ce4b7a 100644 --- a/Sources/BraveWallet/Crypto/Onboarding/LegalView.swift +++ b/Sources/BraveWallet/Crypto/Onboarding/LegalView.swift @@ -76,7 +76,7 @@ struct LegalView: View { .padding() .background( NavigationLink( - destination: CreateWalletContainerView( + destination: CreateWalletView( keyringStore: keyringStore, setupOption: setupOption, dismissAction: dismissAction @@ -89,7 +89,7 @@ struct LegalView: View { ) .background( NavigationLink( - destination: RestoreWalletContainerView( + destination: RestoreWalletView( keyringStore: keyringStore, dismissAction: dismissAction ), diff --git a/Sources/BraveWallet/Crypto/Onboarding/RestoreWalletView.swift b/Sources/BraveWallet/Crypto/Onboarding/RestoreWalletView.swift index 5ce77d65d19..c23461e1bd0 100644 --- a/Sources/BraveWallet/Crypto/Onboarding/RestoreWalletView.swift +++ b/Sources/BraveWallet/Crypto/Onboarding/RestoreWalletView.swift @@ -10,22 +10,7 @@ import Strings import struct Shared.AppConstants import Preferences -struct RestoreWalletContainerView: View { - @ObservedObject var keyringStore: KeyringStore - // Used to dismiss all of Wallet - let dismissAction: () -> Void - - var body: some View { - ScrollView(.vertical) { - RestoreWalletView(keyringStore: keyringStore, dismissAction: dismissAction) - .background(Color(.braveBackground)) - } - .background(Color(.braveBackground).edgesIgnoringSafeArea(.all)) - .transparentUnlessScrolledNavigationAppearance() - } -} - -private struct RestoreWalletView: View { +struct RestoreWalletView: View { @ObservedObject var keyringStore: KeyringStore // Used to dismiss all of Wallet let dismissAction: () -> Void @@ -195,9 +180,13 @@ private struct RestoreWalletView: View { handleRecoveryWordsChanged(oldValue: recoveryWords, newValue: newValue) } .navigationBarBackButtonHidden(isShowingCreatingWallet) + .frame(maxWidth: .infinity, maxHeight: .infinity) + .background(Color(.braveBackground).edgesIgnoringSafeArea(.all)) .overlay { if isShowingCreatingWallet { CreatingWalletView() + .ignoresSafeArea() + .frame(maxWidth: .infinity, maxHeight: .infinity) } } .toolbar(content: { @@ -213,7 +202,7 @@ private struct RestoreWalletView: View { }) .sheet(isPresented: $isShowingCreateNewPassword) { NavigationView { - CreateWalletContainerView( + CreateWalletView( keyringStore: keyringStore, setupOption: .restore, onValidPasswordEntered: restoreWallet, @@ -228,6 +217,7 @@ private struct RestoreWalletView: View { } } } + .transparentUnlessScrolledNavigationAppearance() } private func resignFirstResponder() { @@ -260,7 +250,7 @@ private struct RestoreWalletView: View { struct RestoreWalletView_Previews: PreviewProvider { static var previews: some View { NavigationView { - RestoreWalletContainerView( + RestoreWalletView( keyringStore: .previewStore, dismissAction: {} ) diff --git a/Sources/BraveWallet/Crypto/UnlockWalletView.swift b/Sources/BraveWallet/Crypto/UnlockWalletView.swift index 977021eaf81..f2826052a0d 100644 --- a/Sources/BraveWallet/Crypto/UnlockWalletView.swift +++ b/Sources/BraveWallet/Crypto/UnlockWalletView.swift @@ -82,7 +82,7 @@ struct UnlockWalletView: View { .disabled(!isPasswordValid) NavigationLink( - destination: RestoreWalletContainerView( + destination: RestoreWalletView( keyringStore: keyringStore, dismissAction: dismissAction )