Skip to content
This repository has been archived by the owner on May 10, 2024. It is now read-only.

Commit

Permalink
Update UnlockWalletView to v2 designs
Browse files Browse the repository at this point in the history
  • Loading branch information
StephenHeaps committed Oct 31, 2023
1 parent b14b83e commit e520c04
Show file tree
Hide file tree
Showing 3 changed files with 151 additions and 79 deletions.
218 changes: 145 additions & 73 deletions Sources/BraveWallet/Crypto/UnlockWalletView.swift
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ struct UnlockWalletView: View {
@ObservedObject var keyringStore: KeyringStore

@State private var password: String = ""
@FocusState private var isPasswordFieldFocused: Bool
@State private var unlockError: UnlockError?
@State private var attemptedBiometricsUnlock: Bool = false

Expand All @@ -25,11 +26,18 @@ struct UnlockWalletView: View {
}
}
}

private var isPasswordValid: Bool {
!password.isEmpty
}


private func fillPasswordFromKeychain() {
if let password = keyringStore.retrievePasswordFromKeychain() {
self.password = password
unlock()
}
}

private func unlock() {
// Conflict with the keyboard submit/dismissal that causes a bug
// with SwiftUI animating the screen away...
Expand All @@ -42,101 +50,165 @@ struct UnlockWalletView: View {
}
}
}

private func fillPasswordFromKeychain() {
if let password = keyringStore.retrievePasswordFromKeychain() {
self.password = password
unlock()
}
}

private var biometricsIcon: Image? {
let context = LAContext()
if context.canEvaluatePolicy(.deviceOwnerAuthenticationWithBiometrics, error: nil) {
switch context.biometryType {
case .faceID:
return Image(systemName: "faceid")
case .touchID:
return Image(systemName: "touchid")
case .none:
return nil
@unknown default:
return nil
}
}
return nil
}


var body: some View {
ScrollView(.vertical) {
VStack(spacing: 46) {
Image("graphic-lock", bundle: .module)
.padding(.bottom)
.accessibilityHidden(true)
VStack {
Text(Strings.Wallet.unlockWalletTitle)
.font(.headline)
.padding(.bottom)
.multilineTextAlignment(.center)
.fixedSize(horizontal: false, vertical: true)
HStack {
SecureField(Strings.Wallet.passwordPlaceholder, text: $password, onCommit: unlock)
.textContentType(.password)
.font(.subheadline)
.introspectTextField(customize: { tf in
tf.becomeFirstResponder()
})
.textFieldStyle(BraveValidatedTextFieldStyle(error: unlockError))
if keyringStore.isKeychainPasswordStored, let icon = biometricsIcon {
Button(action: fillPasswordFromKeychain) {
icon
.imageScale(.large)
.font(.headline)
}
ScrollView {
VStack(spacing: 40) {
VStack(spacing: 4) {
Text(Strings.Wallet.unlockWallet)
.font(.title)
.fontWeight(.medium)
.foregroundColor(Color(braveSystemName: .textPrimary))
Text(Strings.Wallet.unlockWalletDescription)
.font(.subheadline)
.foregroundColor(Color(braveSystemName: .textSecondary))
}
.padding(.top, 44)

VStack(spacing: 32) {
SecureField(Strings.Wallet.passwordPlaceholder, text: $password, onCommit: unlock)
.textContentType(.password)
.modifier(WalletUnlockStyleModifier(isFocused: isPasswordFieldFocused, error: unlockError))
.focused($isPasswordFieldFocused)

VStack(spacing: 16) {
Button(action: unlock) {
Text(Strings.Wallet.unlockWalletButtonTitle)
.frame(maxWidth: .infinity)
}
.buttonStyle(BraveFilledButtonStyle(size: .large))
.disabled(!isPasswordValid)

NavigationLink(
destination: RestoreWalletContainerView(
keyringStore: keyringStore
)
) {
Text(Strings.Wallet.restoreWalletButtonTitle)
.fontWeight(.semibold)
.foregroundColor(Color(braveSystemName: .textInteractive))
.padding(.vertical, 10)
.padding(.horizontal, 20)
.frame(maxWidth: .infinity)
}
}
.padding(.horizontal, 48)
}
VStack(spacing: 30) {
Button(action: unlock) {
Text(Strings.Wallet.unlockWalletButtonTitle)
}
.buttonStyle(BraveFilledButtonStyle(size: .normal))
.disabled(!isPasswordValid)
NavigationLink(destination: RestoreWalletContainerView(keyringStore: keyringStore)) {
Text(Strings.Wallet.restoreWalletButtonTitle)
.font(.subheadline.weight(.medium))

if keyringStore.isKeychainPasswordStored, let icon = biometricsIcon {
Button(action: fillPasswordFromKeychain) {
icon
.imageScale(.large)
.font(.headline)
.foregroundColor(Color(braveSystemName: .iconInteractive))
.padding()
.background(Circle()
.strokeBorder(Color(braveSystemName: .dividerInteractive), lineWidth: 1))
}
.foregroundColor(Color(.braveLabel))
}
}
.frame(maxHeight: .infinity, alignment: .top)
.padding()
.padding(.vertical)
.padding(.horizontal, 34)
}
.navigationTitle(Strings.Wallet.cryptoTitle)
.navigationBarTitleDisplayMode(.inline)
.background(Color(.braveBackground).edgesIgnoringSafeArea(.all))
.frame(maxWidth: .infinity, maxHeight: .infinity)
.background(
Image("wallet-background", bundle: .module)
.resizable()
.aspectRatio(contentMode: .fill)
.edgesIgnoringSafeArea(.all)
)
.onChange(of: password) { _ in
unlockError = nil
}
.onAppear {
self.isPasswordFieldFocused = true

DispatchQueue.main.asyncAfter(deadline: .now() + 0.5) { [self] in
if !keyringStore.lockedManually && !attemptedBiometricsUnlock && keyringStore.defaultKeyring.isLocked && UIApplication.shared.isProtectedDataAvailable {
attemptedBiometricsUnlock = true
fillPasswordFromKeychain()
}
}
}
.navigationTitle(Strings.Wallet.cryptoTitle)
.navigationBarTitleDisplayMode(.inline)
.ignoresSafeArea(.keyboard, edges: .bottom)
}

private var biometricsIcon: Image? {
let context = LAContext()
if context.canEvaluatePolicy(.deviceOwnerAuthenticationWithBiometrics, error: nil) {
switch context.biometryType {
case .faceID:
return Image(systemName: "faceid")
case .touchID:
return Image(systemName: "touchid")
case .none:
return nil
@unknown default:
return nil
}
}
return nil
}
}

#if DEBUG
struct CryptoUnlockView_Previews: PreviewProvider {
struct UnlockWalletView_Previews: PreviewProvider {
static var previews: some View {
UnlockWalletView(keyringStore: .previewStore)
.previewLayout(.sizeThatFits)
.previewColorSchemes()
NavigationView {
UnlockWalletView(
keyringStore: .previewStoreWithWalletCreated
)
}
.previewColorSchemes()
}
}
#endif

private struct WalletUnlockStyleModifier<Failure: LocalizedError & Equatable>: ViewModifier {

var isFocused: Bool
var error: Failure?

private var borderColor: Color {
if error != nil {
return Color.red
} else if isFocused {
return Color(braveSystemName: .iconInteractive)
}
return Color.clear
}

func body(content: Content) -> some View {
VStack(spacing: 6) {
content
.padding(.horizontal, 16)
.padding(.vertical, 10)
.background(
RoundedRectangle(cornerRadius: 10, style: .continuous)
.strokeBorder(borderColor, lineWidth: 1)
.background(
RoundedRectangle(cornerRadius: 10, style: .continuous)
.fill(Color(braveSystemName: .containerBackground))
)
)
if let error = error {
HStack(alignment: .firstTextBaseline, spacing: 4) {
Image(braveSystemName: "leo.warning.triangle-outline")
Text(error.localizedDescription)
.fixedSize(horizontal: false, vertical: true)
.animation(nil, value: error.localizedDescription) // Dont animate the text change, just alpha
}
.frame(maxWidth: .infinity, alignment: .leading)
.transition(
.asymmetric(
insertion: .opacity.animation(.default),
removal: .identity
)
)
.font(.footnote)
.foregroundColor(Color(.braveErrorLabel))
.padding(.leading, 8)
}
}
}
}
2 changes: 1 addition & 1 deletion Sources/BraveWallet/Panels/WalletPanelView.swift
Original file line number Diff line number Diff line change
Expand Up @@ -68,7 +68,7 @@ public struct WalletPanelContainerView: View {
} label: {
HStack(spacing: 4) {
Image(braveSystemName: "leo.lock.open")
Text(Strings.Wallet.walletPanelUnlockWallet)
Text(Strings.Wallet.unlockWallet)
}
}
.buttonStyle(BraveFilledButtonStyle(size: .normal))
Expand Down
10 changes: 5 additions & 5 deletions Sources/BraveWallet/WalletStrings.swift
Original file line number Diff line number Diff line change
Expand Up @@ -414,8 +414,8 @@ extension Strings {
value: "Incorrect password",
comment: "The error message displayed when the user enters the wrong password while unlocking the wallet"
)
public static let unlockWalletTitle = NSLocalizedString(
"wallet.unlockWalletTitle",
public static let unlockWalletDescription = NSLocalizedString(
"wallet.unlockWalletDescription",
tableName: "BraveWallet",
bundle: .module,
value: "Enter password to unlock wallet",
Expand Down Expand Up @@ -2802,11 +2802,11 @@ extension Strings {
value: "None",
comment: "The value shown when selecting the default wallet as none / no wallet in wallet settings, or when grouping Portfolio assets."
)
public static let walletPanelUnlockWallet = NSLocalizedString(
"wallet.walletPanelUnlockWallet",
public static let unlockWallet = NSLocalizedString(
"wallet.unlockWallet",
tableName: "BraveWallet",
bundle: .module,
value: "Unlock wallet",
value: "Unlock Wallet",
comment: "The title of the button in wallet panel when wallet is locked. Users can click it to open full screen unlock wallet screen."
)
public static let walletPanelSetupWalletDescription = NSLocalizedString(
Expand Down

0 comments on commit e520c04

Please sign in to comment.