You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
session-ios/Session/Onboarding/LoadAccountScreen.swift

212 lines
7.5 KiB
Swift

// Copyright © 2023 Rangeproof Pty Ltd. All rights reserved.
import SwiftUI
import SessionUIKit
import SignalUtilitiesKit
import SessionUtilitiesKit
import AVFoundation
struct LoadAccountScreen: View {
@EnvironmentObject var host: HostWrapper
@State var tabIndex = 0
@State private var recoveryPassword: String = ""
@State private var hexEncodedSeed: String = ""
@State private var errorString: String? = nil
var body: some View {
ZStack(alignment: .topLeading) {
ThemeManager.currentTheme.colorSwiftUI(for: .backgroundPrimary).ignoresSafeArea()
VStack(
spacing: 0
){
CustomTopTabBar(
tabIndex: $tabIndex,
tabTitles: [
"sessionRecoveryPassword".localized(),
"qrScan".localized()
]
).frame(maxWidth: .infinity)
if tabIndex == 0 {
EnterRecoveryPasswordScreen(
$recoveryPassword,
error: $errorString,
continueWithMnemonic: continueWithMnemonic
)
}
else {
ScanQRCodeScreen(
$hexEncodedSeed,
error: $errorString,
continueAction: continueWithhexEncodedSeed
)
}
}
}
}
private func continueWithSeed(seed: Data, from source: Onboarding.SeedSource, onError: (() -> ())?) {
if (seed.count != 16) {
errorString = source.genericErrorMessage
DispatchQueue.main.asyncAfter(deadline: .now() + 1) {
onError?()
}
return
}
let (ed25519KeyPair, x25519KeyPair) = try! Identity.generate(from: seed)
Onboarding.Flow.link
.preregister(
with: seed,
ed25519KeyPair: ed25519KeyPair,
x25519KeyPair: x25519KeyPair
)
// Otherwise continue on to request push notifications permissions
let viewController: SessionHostingViewController = SessionHostingViewController(rootView: PNModeScreen(flow: .link))
viewController.setUpNavBarSessionIcon()
self.host.controller?.navigationController?.pushViewController(viewController, animated: true)
}
func continueWithhexEncodedSeed(onError: (() -> ())?) {
let seed = Data(hex: hexEncodedSeed)
continueWithSeed(seed: seed, from: .qrCode, onError: onError)
}
func continueWithMnemonic() {
let mnemonic = recoveryPassword.lowercased()
let hexEncodedSeed: String
do {
hexEncodedSeed = try Mnemonic.decode(mnemonic: mnemonic)
} catch {
if let decodingError = error as? Mnemonic.DecodingError {
switch decodingError {
case .inputTooShort, .missingLastWord:
errorString = "recoveryPasswordErrorMessageShort".localized()
case .invalidWord, .verificationFailed:
errorString = "recoveryPasswordErrorMessageIncorrect".localized()
default:
errorString = "recoveryPasswordErrorMessageGeneric".localized()
}
} else {
errorString = "recoveryPasswordErrorMessageGeneric".localized()
}
return
}
let seed = Data(hex: hexEncodedSeed)
continueWithSeed(seed: seed, from: .mnemonic, onError: nil)
}
}
struct EnterRecoveryPasswordScreen: View{
@Binding var recoveryPassword: String
@Binding var error: String?
var continueWithMnemonic: (() -> Void)?
init(
_ recoveryPassword: Binding<String>,
error: Binding<String?>,
continueWithMnemonic: (() -> Void)?
) {
self._recoveryPassword = recoveryPassword
self._error = error
self.continueWithMnemonic = continueWithMnemonic
}
var body: some View{
ZStack(alignment: .center) {
VStack(
alignment: .leading,
spacing: 0
) {
Spacer(minLength: 0)
HStack(
alignment: .bottom,
spacing: Values.smallSpacing
) {
Text("sessionRecoveryPassword".localized())
.bold()
.font(.system(size: Values.veryLargeFontSize))
.foregroundColor(themeColor: .textPrimary)
Image("SessionShield")
.resizable()
.renderingMode(.template)
.foregroundColor(themeColor: .textPrimary)
.scaledToFit()
.frame(
maxWidth: Values.largeFontSize,
maxHeight: Values.largeFontSize
)
.padding(.bottom, Values.verySmallSpacing)
}
Spacer(minLength: 0)
.frame(maxHeight: 2 * Values.mediumSpacing)
Text("onboardingRecoveryPassword".localized())
.font(.system(size: Values.smallFontSize))
.foregroundColor(themeColor: .textPrimary)
.fixedSize(horizontal: false, vertical: true)
Spacer(minLength: 0)
.frame(maxHeight: 2 * Values.mediumSpacing)
SessionTextField(
$recoveryPassword,
placeholder: "recoveryPasswordEnter".localized(),
error: $error,
accessibility: Accessibility(identifier: "Recovery password input")
)
Spacer(minLength: 0)
.frame(maxHeight: Values.massiveSpacing)
Spacer(minLength: 0)
}
.padding(.horizontal, Values.veryLargeSpacing)
.padding(.bottom, Values.largeButtonHeight)
VStack() {
Spacer()
Button {
continueWithMnemonic?()
} label: {
Text("theContinue".localized())
.bold()
.font(.system(size: Values.smallFontSize))
.foregroundColor(themeColor: .sessionButton_text)
.frame(
maxWidth: .infinity,
maxHeight: Values.largeButtonHeight,
alignment: .center
)
.overlay(
Capsule()
.stroke(themeColor: .sessionButton_border)
)
}
.accessibility(
Accessibility(
identifier: "Continue",
label: "Continue"
)
)
.padding(.horizontal, Values.massiveSpacing)
}
.padding(.vertical, Values.mediumSpacing)
}
}
}
struct LoadAccountView_Previews: PreviewProvider {
static var previews: some View {
LoadAccountScreen()
}
}