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/Settings/RecoveryPasswordScreen.swift

306 lines
15 KiB
Swift

// Copyright © 2023 Rangeproof Pty Ltd. All rights reserved.
import SwiftUI
import SessionUIKit
import SignalUtilitiesKit
import SessionUtilitiesKit
struct RecoveryPasswordScreen: View {
@EnvironmentObject var host: HostWrapper
@State private var copied: Bool = false
@State private var showQRCode: Bool = false
private let mnemonic: String
static private let cornerRadius: CGFloat = 13
static private let backgroundCornerRadius: CGFloat = 17
static private let buttonWidth: CGFloat = UIDevice.current.isIPad ? Values.iPadButtonWidth : 130
public init() throws {
self.mnemonic = try Identity.mnemonic()
}
public init(hardcode: String) {
self.mnemonic = hardcode
}
var body: some View {
ZStack(alignment: .center) {
ScrollView(.vertical, showsIndicators: false) {
VStack(
alignment: .leading,
spacing: Values.mediumSpacing
) {
ZStack {
RoundedRectangle(cornerRadius: Self.backgroundCornerRadius)
.fill(themeColor: .backgroundSecondary)
VStack(
alignment: .leading,
spacing: 0
) {
HStack(
alignment: .center,
spacing: Values.smallSpacing
) {
Text("sessionRecoveryPassword".localized())
.bold()
.font(.system(size: Values.mediumFontSize))
.foregroundColor(themeColor: .textPrimary)
Image("SessionShieldFilled")
.resizable()
.renderingMode(.template)
.foregroundColor(themeColor: .textPrimary)
.scaledToFit()
.frame(
maxWidth: Values.mediumFontSize,
maxHeight: Values.mediumFontSize
)
}
Text("recovery_password_explanation_1".localized())
.font(.system(size: Values.smallFontSize))
.foregroundColor(themeColor: .textPrimary)
.padding(.bottom, Values.mediumSpacing)
.fixedSize(horizontal: false, vertical: true)
Text("recovery_password_explanation_2".localized())
.font(.system(size: Values.smallFontSize))
.foregroundColor(themeColor: .textPrimary)
.padding(.bottom, Values.mediumSpacing)
.fixedSize(horizontal: false, vertical: true)
if self.showQRCode {
QRCodeView(
string: mnemonic,
hasBackground: false,
logo: "SessionShieldFilled",
themeStyle: ThemeManager.currentTheme.interfaceStyle
)
.padding(.all, Values.smallSpacing)
ZStack(alignment: .center) {
Button {
self.showQRCode.toggle()
} label: {
Text("recoveryPasswordView".localized())
.bold()
.font(.system(size: Values.verySmallFontSize))
.foregroundColor(themeColor: .textPrimary)
.frame(
maxWidth: Self.buttonWidth,
maxHeight: Values.mediumSmallButtonHeight,
alignment: .center
)
.overlay(
Capsule()
.stroke(themeColor: .textPrimary)
)
}
}
.frame(maxWidth: .infinity)
.padding(.top, Values.mediumSpacing)
} else {
Text(mnemonic)
.font(.spaceMono(size: Values.verySmallFontSize))
.multilineTextAlignment(.center)
.foregroundColor(themeColor: .sessionButton_text)
.frame(
maxWidth: .infinity,
maxHeight: .infinity
)
.fixedSize(horizontal: false, vertical: true)
.padding(.all, Values.largeSpacing)
.overlay(
RoundedRectangle(
cornerSize: CGSize(
width: Self.cornerRadius,
height: Self.cornerRadius
)
)
.stroke(themeColor: .borderSeparator)
)
.padding(.bottom, Values.mediumSpacing)
HStack(
alignment: .center,
spacing: 0
) {
Button {
copyRecoveryPassword()
} label: {
let buttonTitle: String = self.copied ? "copied".localized() : "copy".localized()
Text(buttonTitle)
.bold()
.font(.system(size: Values.verySmallFontSize))
.foregroundColor(themeColor: .textPrimary)
.frame(
maxWidth: .infinity,
minHeight: Values.mediumSmallButtonHeight,
alignment: .center
)
.overlay(
Capsule()
.stroke(themeColor: .textPrimary)
)
}
.frame(maxWidth: .infinity)
Spacer(minLength: Values.veryLargeSpacing)
Button {
self.showQRCode.toggle()
} label: {
Text("qrView".localized())
.bold()
.font(.system(size: Values.verySmallFontSize))
.foregroundColor(themeColor: .textPrimary)
.frame(
maxWidth: .infinity,
minHeight: Values.mediumSmallButtonHeight,
alignment: .center
)
.overlay(
Capsule()
.stroke(themeColor: .textPrimary)
)
}
.frame(maxWidth: .infinity)
}
}
}
.padding(.all, Values.mediumSpacing)
}
ZStack {
RoundedRectangle(cornerRadius: Self.backgroundCornerRadius)
.fill(themeColor: .backgroundSecondary)
HStack(
alignment: .center,
spacing: Values.mediumSpacing
) {
VStack(
alignment: .leading,
spacing: 0
) {
Text("recoveryPasswordHideRecoveryPassword".localized())
.bold()
.font(.system(size: Values.mediumFontSize))
.foregroundColor(themeColor: .textPrimary)
Text("recoveryPasswordHideRecoveryPasswordDescription".localized())
.font(.system(size: Values.smallFontSize))
.foregroundColor(themeColor: .textPrimary)
}
.frame(
maxWidth: .infinity,
alignment: .leading
)
Button {
hideRecoveryPassword()
} label: {
Text("hide".localized())
.bold()
.font(.system(size: Values.verySmallFontSize))
.foregroundColor(themeColor: .danger)
.frame(
width: 55,
height: Values.mediumSmallButtonHeight
)
.overlay(
Capsule()
.stroke(themeColor: .danger)
)
}
.accessibility(
Accessibility(
identifier: "Hide recovery password button",
label: "Hide recovery password button"
)
)
}
.padding(.all, Values.mediumSpacing)
}
}
.padding(.horizontal, Values.largeSpacing)
.padding(.vertical, Values.mediumSpacing)
}
}
.backgroundColor(themeColor: .backgroundPrimary)
.onAppear {
Storage.shared.writeAsync { db in db[.hasViewedSeed] = true }
}
}
private func copyRecoveryPassword() {
UIPasteboard.general.string = self.mnemonic
self.copied = true
}
private func hideRecoveryPassword() {
let modal: ConfirmationModal = ConfirmationModal(
info: ConfirmationModal.Info(
title: "recoveryPasswordHidePermanently".localized(),
body: .text(
"hide_recovery_password_modal_warning_1".localized() +
"\n\n" +
"hide_recovery_password_modal_warning_2".localized()
),
confirmTitle: "continue_2".localized(),
confirmAccessibility: Accessibility(
identifier: "Continue",
label: "Continue"
),
confirmStyle: .danger,
cancelAccessibility: Accessibility(
identifier: "Cancel",
label: "Cancel"
),
cancelStyle: .textPrimary,
onConfirm: { modal in
guard let presentingViewController: UIViewController = modal.presentingViewController else {
return
}
let continueModal: ConfirmationModal = ConfirmationModal(
info: ConfirmationModal.Info(
title: "recoveryPasswordHidePermanently".localized(),
body: .text("recoveryPasswordHidePermanentlyDescription2".localized()),
confirmTitle: "TXT_CANCEL_TITLE".localized(),
confirmAccessibility: Accessibility(
identifier: "Cancel",
label: "Cancel"
),
confirmStyle: .textPrimary,
cancelTitle: "yes".localized(),
cancelAccessibility: Accessibility(
identifier: "Confirm button",
label: "Confirm button"
),
cancelStyle: .danger,
onCancel: { modal in
modal.dismiss(animated: true) {
Storage.shared.writeAsync { db in db[.hideRecoveryPasswordPermanently] = true }
self.host.controller?.navigationController?.popViewController(animated: true)
}
}
)
)
return presentingViewController.present(continueModal, animated: true, completion: nil)
}
)
)
self.host.controller?.present(modal, animated: true)
}
}
struct RecoveryPasswordView_Previews: PreviewProvider {
static var previews: some View {
RecoveryPasswordScreen(hardcode: "Voyage urban toyed maverick peculiar tuxedo penguin tree grass building listen speak withdraw terminal plane")
}
}