From 3a7c2eb0143fdc939909a11d01c5c2e38b4a97c7 Mon Sep 17 00:00:00 2001 From: ant013 Date: Tue, 3 Oct 2023 19:41:28 +0400 Subject: [PATCH] Inmplement Backup App Settings module --- .../project.pbxproj | 140 +++++++ .../Icons/file_24.imageset/Contents.json | 22 + .../Icons/file_24.imageset/file@2x.png | Bin 0 -> 472 bytes .../Icons/file_24.imageset/file@3x.png | Bin 0 -> 702 bytes .../Core/Managers/CloudBackupManager.swift | 21 +- .../Core/Managers/EvmSyncSourceManager.swift | 10 +- .../BackupCloudPassphraseViewModel.swift | 37 +- .../BottomSheet/BottomSheetModule.swift | 18 + .../Settings/BackupApp/BackupAppModule.swift | 52 +++ .../BackupApp/BackupAppViewModel.swift | 379 ++++++++++++++++++ .../BackupDisclaimerView.swift | 53 +++ .../BackupApp/BackupList/BackupListView.swift | 104 +++++ .../BackupManager/BackupManagerModule.swift | 7 + .../BackupManager/BackupManagerView.swift | 35 ++ .../BackupApp/BackupName/BackupNameView.swift | 52 +++ .../BackupPassword/BackupPasswordView.swift | 109 +++++ .../BackupApp/BackupType/BackupTypeView.swift | 69 ++++ .../BlockchainSettingsViewModel.swift | 2 +- .../Main/MainSettingsViewController.swift | 13 +- .../UserInterface/Caution.swift | 38 +- .../UserInterface/SwiftUI/CheckboxStyle.swift | 23 ++ .../Extensions/ActivityViewController.swift | 22 + .../SwiftUI/Extensions/Text.swift | 27 +- .../UserInterface/SwiftUI/InputTextRow.swift | 15 + .../UserInterface/SwiftUI/InputTextView.swift | 125 ++++++ .../UserInterface/SwiftUI/NavigationRow.swift | 15 +- .../UserInterface/SwiftUI/Shake.swift | 86 ++++ .../en.lproj/Localizable.strings | 42 ++ 28 files changed, 1475 insertions(+), 41 deletions(-) create mode 100644 UnstoppableWallet/UnstoppableWallet/Assets.xcassets/Icons/file_24.imageset/Contents.json create mode 100644 UnstoppableWallet/UnstoppableWallet/Assets.xcassets/Icons/file_24.imageset/file@2x.png create mode 100644 UnstoppableWallet/UnstoppableWallet/Assets.xcassets/Icons/file_24.imageset/file@3x.png create mode 100644 UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupAppModule.swift create mode 100644 UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupAppViewModel.swift create mode 100644 UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupDisclaimer/BackupDisclaimerView.swift create mode 100644 UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupList/BackupListView.swift create mode 100644 UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupManager/BackupManagerModule.swift create mode 100644 UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupManager/BackupManagerView.swift create mode 100644 UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupName/BackupNameView.swift create mode 100644 UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupPassword/BackupPasswordView.swift create mode 100644 UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupType/BackupTypeView.swift create mode 100644 UnstoppableWallet/UnstoppableWallet/UserInterface/SwiftUI/CheckboxStyle.swift create mode 100644 UnstoppableWallet/UnstoppableWallet/UserInterface/SwiftUI/Extensions/ActivityViewController.swift create mode 100644 UnstoppableWallet/UnstoppableWallet/UserInterface/SwiftUI/InputTextRow.swift create mode 100644 UnstoppableWallet/UnstoppableWallet/UserInterface/SwiftUI/InputTextView.swift create mode 100644 UnstoppableWallet/UnstoppableWallet/UserInterface/SwiftUI/Shake.swift diff --git a/UnstoppableWallet/UnstoppableWallet.xcodeproj/project.pbxproj b/UnstoppableWallet/UnstoppableWallet.xcodeproj/project.pbxproj index 345923896a..ba686d0d9f 100644 --- a/UnstoppableWallet/UnstoppableWallet.xcodeproj/project.pbxproj +++ b/UnstoppableWallet/UnstoppableWallet.xcodeproj/project.pbxproj @@ -1994,6 +1994,7 @@ ABC9A05D9F96BE464CFC90CC /* ContactBookModule.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A5518367F0DDDB94D320 /* ContactBookModule.swift */; }; ABC9A06BB934A43890376A70 /* NftCollectionCellFactory.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A8CF40E995105E7F38AC /* NftCollectionCellFactory.swift */; }; ABC9A06BE632BD33E5CA4106 /* Contact.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AEAD18F73D4FBE05783D /* Contact.swift */; }; + ABC9A06CA0D364D4E1261C87 /* BackupPasswordView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AEC2C4752498AF7A7A2E /* BackupPasswordView.swift */; }; ABC9A074995C051E714FAFAB /* BackupContact.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AC50307ABA3DF7034E1D /* BackupContact.swift */; }; ABC9A07518E5769122DFEAC2 /* RestoreCloudViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AA751C8B09F90F716231 /* RestoreCloudViewController.swift */; }; ABC9A0779C1107119CD3AF19 /* FileManager.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A8CE84FA36438BE4D6B5 /* FileManager.swift */; }; @@ -2003,6 +2004,7 @@ ABC9A092DC0DEEF9838DB47A /* CellElement.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A776346AF62265896CA1 /* CellElement.swift */; }; ABC9A096B05E5491A40A327C /* DonateDescriptionDataSource.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9ABFE62D22F9FB0B3409A /* DonateDescriptionDataSource.swift */; }; ABC9A097A0BDD99777D5374D /* DonateDescriptionCell.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AD0848221B0EC25C37F3 /* DonateDescriptionCell.swift */; }; + ABC9A09E0B614E5B4E32B7F9 /* InputTextView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9ACEC3169A9F01B55921A /* InputTextView.swift */; }; ABC9A0A3A52AD41643D67D3D /* SingleLineFormTextView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A03401172C4C65D66764 /* SingleLineFormTextView.swift */; }; ABC9A0AADAE0A5C370946B8D /* RestoreCloudPassphraseViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AA99463E646706E8E36D /* RestoreCloudPassphraseViewModel.swift */; }; ABC9A0B37693470DAD0FFE20 /* WalletConnectMainService.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AD18F3E73F96DD6C4FA9 /* WalletConnectMainService.swift */; }; @@ -2016,7 +2018,9 @@ ABC9A0EE5E5B31405569BF3F /* IndicatorAdviceCell.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AACC40370E1E0CFC7639 /* IndicatorAdviceCell.swift */; }; ABC9A0F42A6687705CAD1340 /* NftAssetViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AF9C0D0174A5B6A91F13 /* NftAssetViewController.swift */; }; ABC9A1117A41AB8CE00FDEDB /* WalletConnectAppShowView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A845B2969166028BA5F0 /* WalletConnectAppShowView.swift */; }; + ABC9A12A4D114A2E4F4C711A /* ActivityViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AF395EA01B43D6D77C43 /* ActivityViewController.swift */; }; ABC9A133A6BF0FC9A87FA14A /* ContactBookSettingsViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A99184EE1D5D052C52E9 /* ContactBookSettingsViewController.swift */; }; + ABC9A13B7F43722709570161 /* BackupAppViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A7B2F7354F12F78A2F30 /* BackupAppViewModel.swift */; }; ABC9A13D78DD5F176A170B65 /* FullBackup.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A41F6AA0B65FDA91EB68 /* FullBackup.swift */; }; ABC9A13DB3ADB580D59F66E4 /* SendEip1155ViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A23CB332521C0607CC6B /* SendEip1155ViewModel.swift */; }; ABC9A13F4C814FFB31FF13CA /* SendEip721ViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A7315E119F0B1581B70C /* SendEip721ViewController.swift */; }; @@ -2033,7 +2037,9 @@ ABC9A1E5E31DD2F72BB2A13A /* IntegerAmountInputViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A6D56EBB7FFAD68CFD66 /* IntegerAmountInputViewModel.swift */; }; ABC9A1EC656488FF79F458EC /* RestoreCloudViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A5FE0EDA53E4D9B85DE1 /* RestoreCloudViewModel.swift */; }; ABC9A1FFFB4F9EC58BF78661 /* AccountRestoreWarningManager.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A7D665A025E95697C757 /* AccountRestoreWarningManager.swift */; }; + ABC9A2035980B70E1C0790A8 /* CheckboxStyle.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AE522F09C5E7029CA86E /* CheckboxStyle.swift */; }; ABC9A20A25C4C683A73CB994 /* ContactBookContactViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A8080797194017F736AB /* ContactBookContactViewModel.swift */; }; + ABC9A20B7F90F5E741A74823 /* BackupAppViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A7B2F7354F12F78A2F30 /* BackupAppViewModel.swift */; }; ABC9A20D2DDF8736293DE5C5 /* CoinIndicatorViewItemFactory.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A76776AD840DBFAA1804 /* CoinIndicatorViewItemFactory.swift */; }; ABC9A20F6F7D5EA2A1A55A9E /* ContactLabelService.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AB89F64056FFB98928E7 /* ContactLabelService.swift */; }; ABC9A227648FF076E9518703 /* ContactBookHelper.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AE12A5E8B9FB24FFE42F /* ContactBookHelper.swift */; }; @@ -2079,6 +2085,7 @@ ABC9A359DB8C1A89269236CC /* ContactBookSettingsModule.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A24CBB826A2D2F88EC61 /* ContactBookSettingsModule.swift */; }; ABC9A3613C5477C07F37F48C /* WalletConnectListViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AE5FD79ECC4AC85B86FA /* WalletConnectListViewController.swift */; }; ABC9A36297D869E49C152CAB /* SwapRevokeConfirmationViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A1360FE305343B1049CF /* SwapRevokeConfirmationViewController.swift */; }; + ABC9A36D3A4EEABF6EA6DBA0 /* Shake.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A104D916039D690E454E /* Shake.swift */; }; ABC9A395A96C1F7C30F21940 /* WalletConnectPendingRequestsService.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AD3F677671FB57CCD886 /* WalletConnectPendingRequestsService.swift */; }; ABC9A3B155B3F6E7E0F2CB07 /* HudHelper.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A381CB4C09FF7CB62A94 /* HudHelper.swift */; }; ABC9A3BC9A18F74818EF5C17 /* MetadataMonitor.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AA2491ADC4E5E089CD42 /* MetadataMonitor.swift */; }; @@ -2105,20 +2112,26 @@ ABC9A47D4666FA5115F98629 /* ChartIndicatorsRepository.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A3758FE2D56036DF27FF /* ChartIndicatorsRepository.swift */; }; ABC9A4801E4964F6AED1E667 /* WalletConnectMainViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AEA1D717D8CED8462AB0 /* WalletConnectMainViewModel.swift */; }; ABC9A4929EFBFAD0B595A4E8 /* RestoreCloudModule.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A45E29D1773EF27A0074 /* RestoreCloudModule.swift */; }; + ABC9A4A21CFBA188A7EEC930 /* ActivityViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AF395EA01B43D6D77C43 /* ActivityViewController.swift */; }; ABC9A4B643D98FB95F431401 /* SendBitcoinAmountInputService.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9ACF1F55164BDFD049793 /* SendBitcoinAmountInputService.swift */; }; ABC9A4B9A4CC3A9EE9A89C32 /* EventHandler.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A950663B76424B1761B3 /* EventHandler.swift */; }; ABC9A4BD4CA7A7872CE6167E /* BaseSendViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9ABE97578DC667CBDC11A /* BaseSendViewController.swift */; }; ABC9A4CD35CF43C88EC13909 /* SendEip1155Service.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AC6A0B950C0AABD5A93E /* SendEip1155Service.swift */; }; ABC9A4D5326B3A85888140FE /* ChartIndicatorSettingsViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9ACE2CCBDF21572F5600C /* ChartIndicatorSettingsViewModel.swift */; }; ABC9A4E323FF7AAD86FA8E75 /* MarketCardView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A37521CD6E2CC5BA4E68 /* MarketCardView.swift */; }; + ABC9A4EE80A57455DB2CCD4F /* BackupDisclaimerView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9ABF9E55990704693487F /* BackupDisclaimerView.swift */; }; ABC9A4F4B7F17169DC240A98 /* WalletConnectUriHandler.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A1C31F5343EB2BEA4540 /* WalletConnectUriHandler.swift */; }; ABC9A4FF1E1964FB77700C4E /* ChartIndicatorFactory.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A6DE5C760A5D0C90B70E /* ChartIndicatorFactory.swift */; }; + ABC9A50F1811EEABE61913F7 /* BackupManagerView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A121EDBB1FF651D514D1 /* BackupManagerView.swift */; }; + ABC9A5125874E09194BA8532 /* BackupTypeView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AC9C3F666BC8D15C7F53 /* BackupTypeView.swift */; }; ABC9A51979D2047BEF45A2AE /* TokenSelectView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A791A47F4F1E71B51B3B /* TokenSelectView.swift */; }; ABC9A51E36466E414AF24C67 /* WalletConnectMainPendingRequestViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A8D8072033A5AC7E4897 /* WalletConnectMainPendingRequestViewModel.swift */; }; ABC9A51F2E7EB0B477EBE708 /* SendZcashService.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AE6D2CD14194802E7976 /* SendZcashService.swift */; }; ABC9A52E08E5C57665C07DBC /* PseudoAccessoryView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AEC034DE5784F55BD5F3 /* PseudoAccessoryView.swift */; }; + ABC9A542CA987F09C93F04A9 /* InputTextRow.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A7FA830E64B8DCA1A69A /* InputTextRow.swift */; }; ABC9A543EB59D153FAD103F6 /* KdfParams.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A6663522498A53CF4174 /* KdfParams.swift */; }; ABC9A54917CDA7F9EAE237C4 /* ChartIndicatorsModule.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AB3EC7A1FB0D6C9F7F89 /* ChartIndicatorsModule.swift */; }; + ABC9A54E6E15E96271525339 /* BackupPasswordView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AEC2C4752498AF7A7A2E /* BackupPasswordView.swift */; }; ABC9A54FFFFBFC3C7B23F0B8 /* BottomGradientHolder.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9ADE822BC024F9B798211 /* BottomGradientHolder.swift */; }; ABC9A55470228BD7B1535B9B /* WalletTokenBalanceViewItemFactory.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A86EA911DA12C7A6AC20 /* WalletTokenBalanceViewItemFactory.swift */; }; ABC9A556361B2644555659D8 /* SendConfirmationViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AA7F2ECF212EF8B70470 /* SendConfirmationViewController.swift */; }; @@ -2156,6 +2169,7 @@ ABC9A6A792282ACC8DAB62BC /* IntegerFormAmountInputView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AB9077A6A0ABE4909B76 /* IntegerFormAmountInputView.swift */; }; ABC9A6A9C28C95352232B062 /* ThemeMode.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A830FE79DBF62FD63CC4 /* ThemeMode.swift */; }; ABC9A6BC79804B3D3AAFA8F1 /* SendZcashService.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AE6D2CD14194802E7976 /* SendZcashService.swift */; }; + ABC9A6C05017D9267567AB73 /* BackupDisclaimerView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9ABF9E55990704693487F /* BackupDisclaimerView.swift */; }; ABC9A6C0A45A33C83B632D58 /* SendFeeService.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9ADB77831DCB474B24C8A /* SendFeeService.swift */; }; ABC9A6C1B2F55F1FFA8910CA /* ContactBookSettingsViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A06A4A02C5E889265463 /* ContactBookSettingsViewModel.swift */; }; ABC9A6C65416E7F4F3830962 /* RestoreCloudPassphraseViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AF12879C62002DFE946A /* RestoreCloudPassphraseViewController.swift */; }; @@ -2165,6 +2179,7 @@ ABC9A6F88E51293F2605CACD /* ContactBookContactModule.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AB2ED4E48D4FCEDBE769 /* ContactBookContactModule.swift */; }; ABC9A70AE588307EA1D3A414 /* SendConfirmationService.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A2FD4A59DB53631435BA /* SendConfirmationService.swift */; }; ABC9A712F6389F5C2B0D63E3 /* RestoreCloudService.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A06866150862CEDEB5DE /* RestoreCloudService.swift */; }; + ABC9A7244B2D23BDA40C407C /* BackupNameView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A86FE7E7F2DA93A5CB00 /* BackupNameView.swift */; }; ABC9A727EC8A3C74C9C1A31A /* WalletConnectMainViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A3A694467493C6F4AACE /* WalletConnectMainViewController.swift */; }; ABC9A728EEF4A054C7B8722B /* DonateAddressModule.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A0B7E7360DC0357B2D0F /* DonateAddressModule.swift */; }; ABC9A7297650FD2D9F8F595F /* MacdIndicatorDataSource.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AA8F31619609907AD67E /* MacdIndicatorDataSource.swift */; }; @@ -2177,12 +2192,15 @@ ABC9A7655AE66379E42FE2A4 /* ContactBookSettingsViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A99184EE1D5D052C52E9 /* ContactBookSettingsViewController.swift */; }; ABC9A767A49B686B3A3AC154 /* UniswapV3Module.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A6363DB5DAE5B58AFDC0 /* UniswapV3Module.swift */; }; ABC9A774500F8D8D3D9E04DD /* SendBitcoinAdapterService.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A1BD3B1B53C72DDF923A /* SendBitcoinAdapterService.swift */; }; + ABC9A77A41517173B610E0FC /* BackupManagerModule.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AD8AEA1E3B1198DDECCE /* BackupManagerModule.swift */; }; ABC9A78CFF8B232D330EC7B5 /* DiffLabel.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A916C64B5EA9D96B8FDA /* DiffLabel.swift */; }; ABC9A78D3A4267CAC0F5D0E8 /* SendConfirmationModule.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AFF7119B9AC0E32B2060 /* SendConfirmationModule.swift */; }; ABC9A794E47FC07ABFC32BBD /* FeePriceScale.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AF8E8DE67732371A00E0 /* FeePriceScale.swift */; }; + ABC9A79CFCEBAC442A1B791D /* BackupAppModule.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A37065F4A8459C416F0A /* BackupAppModule.swift */; }; ABC9A7A9053C6ECF618D0E4A /* WalletConnectSession.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A4544AB5CA22ADE16417 /* WalletConnectSession.swift */; }; ABC9A7A9E27CC5F93BE5018B /* WalletConnectListService.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A3BEB33F6DBE2395FD11 /* WalletConnectListService.swift */; }; ABC9A7AF4EE29CDE045ADEF7 /* MarketCategoryMarketCapFetcher.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9ADC1A3B17225B6CC0869 /* MarketCategoryMarketCapFetcher.swift */; }; + ABC9A7C2087C3A641C3F9AD4 /* Shake.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A104D916039D690E454E /* Shake.swift */; }; ABC9A7CBFDC0DF741E29EA44 /* Integer.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AF26FDCB363793BF66E1 /* Integer.swift */; }; ABC9A7E1F93B0A85976C826D /* UniswapV3Provider.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A253877D9FB972EFB8D7 /* UniswapV3Provider.swift */; }; ABC9A7E28714A9A19A2160D4 /* SendModule.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AD1F2311CC6425CF9D90 /* SendModule.swift */; }; @@ -2192,6 +2210,7 @@ ABC9A806CB34CB9A5E27A0A3 /* RestoreCloudPassphraseViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AF12879C62002DFE946A /* RestoreCloudPassphraseViewController.swift */; }; ABC9A80BCDA72347C6619E6C /* SendTimeLockErrorService.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9ADF114FCFABEA148AF04 /* SendTimeLockErrorService.swift */; }; ABC9A819DDAEE683FCCA02EF /* NftAssetCellFactory.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A30A8F78E9C9AEE861F1 /* NftAssetCellFactory.swift */; }; + ABC9A81D09F820C68D4F75FB /* BackupManagerModule.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AD8AEA1E3B1198DDECCE /* BackupManagerModule.swift */; }; ABC9A82D771D920162551294 /* WalletConnectPendingRequestsViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AE15C187118DE6F0CE7B /* WalletConnectPendingRequestsViewModel.swift */; }; ABC9A83233DA4C83AF83E483 /* WalletConnectPendingRequestsService.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AD3F677671FB57CCD886 /* WalletConnectPendingRequestsService.swift */; }; ABC9A8451CEF02EA0A94CEAA /* ProFeaturesAuthorizationManager.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A9628A708749A31EEA70 /* ProFeaturesAuthorizationManager.swift */; }; @@ -2209,6 +2228,7 @@ ABC9A89499016C8AC8341238 /* NftCollectionCellFactory.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A8CF40E995105E7F38AC /* NftCollectionCellFactory.swift */; }; ABC9A8A74C527C4E01EBB8A5 /* RestoreCloudModule.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A45E29D1773EF27A0074 /* RestoreCloudModule.swift */; }; ABC9A8AE39B8925B28B97F77 /* AppBackupProvider.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AB61EA3B39D8BDB1EEDE /* AppBackupProvider.swift */; }; + ABC9A8CAD0A3AAB7398C2A30 /* BackupManagerView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A121EDBB1FF651D514D1 /* BackupManagerView.swift */; }; ABC9A8CBDB7CF4E781896C49 /* RestoreTypeModule.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AAC741F9A54293CD21B1 /* RestoreTypeModule.swift */; }; ABC9A8D215CC5D6A70736E84 /* SendBaseService.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A48552CF0C90E22686A9 /* SendBaseService.swift */; }; ABC9A8D8709EC2B40D74A97A /* SwapRevokeConfirmationViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A1360FE305343B1049CF /* SwapRevokeConfirmationViewController.swift */; }; @@ -2221,6 +2241,7 @@ ABC9A92D7F9ADCE00CBCED09 /* WalletConnectPairingViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AE62C0399849EFB5C158 /* WalletConnectPairingViewModel.swift */; }; ABC9A933C2603486BA181B19 /* SendFeeService.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9ADB77831DCB474B24C8A /* SendFeeService.swift */; }; ABC9A9493F250B81E1152012 /* SendBitcoinService.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A696DCBBE4761E77311C /* SendBitcoinService.swift */; }; + ABC9A94A82E55F437BD3FE68 /* BackupNameView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A86FE7E7F2DA93A5CB00 /* BackupNameView.swift */; }; ABC9A9562DD283B6FCACBCF9 /* MarketCardTitleView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AC7983E4A81E421AB639 /* MarketCardTitleView.swift */; }; ABC9A95E667DD7BD26602D8E /* SendEip721Service.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9ACE7CB7CC9C118C72559 /* SendEip721Service.swift */; }; ABC9A96132AD85DD613EC773 /* ProFeaturesStorage.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AB785128005F6C2C9F9A /* ProFeaturesStorage.swift */; }; @@ -2228,6 +2249,7 @@ ABC9A99724D817AF0E6C5EC3 /* FileStorage.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AB0A37663BC3F17C7A81 /* FileStorage.swift */; }; ABC9A99861B1F83A19EA370D /* SettingsBackup.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AA7FC181E0E0FB74BEF5 /* SettingsBackup.swift */; }; ABC9A998ECDE5438D94FBAE7 /* MarketDiscoveryCategoryService.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9ADFD9DA59BD2FB21C51B /* MarketDiscoveryCategoryService.swift */; }; + ABC9A99A45187C36D48840F8 /* BackupAppModule.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A37065F4A8459C416F0A /* BackupAppModule.swift */; }; ABC9A9A9FE5A83A6F0C3BFE9 /* SendEip721ViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A7C3BC5FC664BBF14C4F /* SendEip721ViewModel.swift */; }; ABC9A9AC7890BE4AAE7DDC84 /* WalletConnectSessionManager.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AC0B5943DF3B61B20BF6 /* WalletConnectSessionManager.swift */; }; ABC9A9CDDC14BA6259450ECA /* WalletConnectPairingModule.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A4BA46EDEEAB6CD9B25C /* WalletConnectPairingModule.swift */; }; @@ -2235,6 +2257,7 @@ ABC9A9E3191338CD0D1DE8AE /* WalletConnectModule.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A0F966294A4E629CCB65 /* WalletConnectModule.swift */; }; ABC9A9EBBC60A709836DE237 /* NftAssetViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AF9C0D0174A5B6A91F13 /* NftAssetViewController.swift */; }; ABC9A9FA3285B39D25801C2A /* AccountRestoreWarningManager.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A7D665A025E95697C757 /* AccountRestoreWarningManager.swift */; }; + ABC9AA05527830EFFCFB98E3 /* BackupListView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AA0021E969A73DA7E177 /* BackupListView.swift */; }; ABC9AA0E63188150CD9A8D03 /* RestoreTypeViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A13DB598B22516E5AD76 /* RestoreTypeViewModel.swift */; }; ABC9AA27A42D7E2A72B4A932 /* RestoreTypeModule.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AAC741F9A54293CD21B1 /* RestoreTypeModule.swift */; }; ABC9AA27A709AC5F85176A53 /* WalletConnectModule.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A0F966294A4E629CCB65 /* WalletConnectModule.swift */; }; @@ -2261,6 +2284,7 @@ ABC9AB3DAD30AA400DEB719C /* SendBitcoinService.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A696DCBBE4761E77311C /* SendBitcoinService.swift */; }; ABC9AB401FD98F99EF6B07C6 /* RestoreTypeViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A939DD222D4A2BD3D71C /* RestoreTypeViewController.swift */; }; ABC9AB4DF4CCEA2C51DD4AB6 /* SingleLineFormTextView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A03401172C4C65D66764 /* SingleLineFormTextView.swift */; }; + ABC9AB65B10FDB5F30E2731D /* BackupTypeView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AC9C3F666BC8D15C7F53 /* BackupTypeView.swift */; }; ABC9AB6EB596E2F8B15D00E4 /* CipherParams.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A89726499CDB4F697EDD /* CipherParams.swift */; }; ABC9AB71563E5F2C9F2EA9E4 /* WalletConnectAppShowViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A3FB680357E569B6DB5F /* WalletConnectAppShowViewModel.swift */; }; ABC9AB83EE3F909BD80E0539 /* BackupCryptoHelper.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AD5CB1911A698718213F /* BackupCryptoHelper.swift */; }; @@ -2276,6 +2300,7 @@ ABC9ABD9B19AD5D97E332EBE /* SendBinanceViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AF15BD67548E6D755CA0 /* SendBinanceViewController.swift */; }; ABC9ABE2B6B19113D7C5EDA3 /* ContactBookHelper.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AE12A5E8B9FB24FFE42F /* ContactBookHelper.swift */; }; ABC9ABE3189E497EC732B331 /* BackupCloudPassphraseViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A5544C221860C10BF131 /* BackupCloudPassphraseViewModel.swift */; }; + ABC9ABE3F52BF2307533D8FB /* InputTextRow.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A7FA830E64B8DCA1A69A /* InputTextRow.swift */; }; ABC9ABF97B8725530463FBCF /* NftAssetOverviewViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AB8907B0E779CA4DF8F1 /* NftAssetOverviewViewModel.swift */; }; ABC9ABF99296DEA24FC5BFF0 /* SendAmountCautionService.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A580220B9FD291A6496A /* SendAmountCautionService.swift */; }; ABC9ABFA7299ADDDFEE918F7 /* WalletConnectPairingViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9ACE88105815BFC477D71 /* WalletConnectPairingViewController.swift */; }; @@ -2313,6 +2338,7 @@ ABC9ACDFA2F5F3BD9517723D /* NftAssetOverviewViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A806FD17A129212E3F7C /* NftAssetOverviewViewController.swift */; }; ABC9ACE1EDEA27A054EDC2C4 /* ContactBookService.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AC4A19838CA08603E17B /* ContactBookService.swift */; }; ABC9ACE255480B2D6E340611 /* ChartIndicatorsService.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A2F3E5147E0E92258FBB /* ChartIndicatorsService.swift */; }; + ABC9ACE31EC94ABC5B325693 /* BackupListView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AA0021E969A73DA7E177 /* BackupListView.swift */; }; ABC9ACEE45E455BA098231EE /* SendMemoInputCell.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A3C103C1DE359184D944 /* SendMemoInputCell.swift */; }; ABC9AD05E7B986179310D6D7 /* SwapInputAccessoryView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A56611CF5E7B3F25CD5C /* SwapInputAccessoryView.swift */; }; ABC9AD1C8D0CE88A604D5250 /* SendBinanceFactory.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AD0DD32AB4B9BAB79F11 /* SendBinanceFactory.swift */; }; @@ -2355,6 +2381,7 @@ ABC9AE44EF7D6B6419955B9B /* SendEip721Service.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9ACE7CB7CC9C118C72559 /* SendEip721Service.swift */; }; ABC9AE472D9105F6FDAECD42 /* BackupCloudPassphraseService.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AF6AA02DA39787C053F0 /* BackupCloudPassphraseService.swift */; }; ABC9AE4FD599490B0A23003D /* PredefinedBlockchainService.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A1057AD189DA1CE31BF5 /* PredefinedBlockchainService.swift */; }; + ABC9AE51262C09EABF5CCEEE /* InputTextView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9ACEC3169A9F01B55921A /* InputTextView.swift */; }; ABC9AE553D422A163A09E5F8 /* MarketCardValueView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A4FCDC5085002DF35C17 /* MarketCardValueView.swift */; }; ABC9AE6D877341985A6F651F /* SendBitcoinAmountInputService.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9ACF1F55164BDFD049793 /* SendBitcoinAmountInputService.swift */; }; ABC9AE775BB25CDB7AA83228 /* EventHandler.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A950663B76424B1761B3 /* EventHandler.swift */; }; @@ -2385,6 +2412,7 @@ ABC9AF77EF53B4A7B0C0E55A /* SendAmountCautionViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A07A33870908ED1BA338 /* SendAmountCautionViewModel.swift */; }; ABC9AF8136B78C2F3E66FF23 /* NftAssetOverviewViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A806FD17A129212E3F7C /* NftAssetOverviewViewController.swift */; }; ABC9AF81A6F30F0041FE1FAC /* ContactBookAddressViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A55B0E99C1DD25839EDB /* ContactBookAddressViewController.swift */; }; + ABC9AF95141EA649524FBF88 /* CheckboxStyle.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AE522F09C5E7029CA86E /* CheckboxStyle.swift */; }; ABC9AF9C828BEBB740468204 /* WalletBackup.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9AECEEB35D57CB0965E79 /* WalletBackup.swift */; }; ABC9AF9D42B60D05030D43F3 /* ChartIndicatorsViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A0C131342CC764890C2B /* ChartIndicatorsViewController.swift */; }; ABC9AF9F8113DB5D54140E7A /* SendBitcoinViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = ABC9A7D6C9D12C1F1F3A1218 /* SendBitcoinViewController.swift */; }; @@ -3763,9 +3791,11 @@ ABC9A0B7E7360DC0357B2D0F /* DonateAddressModule.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = DonateAddressModule.swift; sourceTree = ""; }; ABC9A0C131342CC764890C2B /* ChartIndicatorsViewController.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = ChartIndicatorsViewController.swift; sourceTree = ""; }; ABC9A0F966294A4E629CCB65 /* WalletConnectModule.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = WalletConnectModule.swift; sourceTree = ""; }; + ABC9A104D916039D690E454E /* Shake.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = Shake.swift; sourceTree = ""; }; ABC9A1057AD189DA1CE31BF5 /* PredefinedBlockchainService.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = PredefinedBlockchainService.swift; sourceTree = ""; }; ABC9A10A83A43DCAFA709472 /* CoinDetailAdviceViewController.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = CoinDetailAdviceViewController.swift; sourceTree = ""; }; ABC9A1136889E6976E17B347 /* WalletConnectService.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = WalletConnectService.swift; sourceTree = ""; }; + ABC9A121EDBB1FF651D514D1 /* BackupManagerView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = BackupManagerView.swift; sourceTree = ""; }; ABC9A12529DC8DE5D46D9776 /* ContactBookAddressViewModel.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = ContactBookAddressViewModel.swift; sourceTree = ""; }; ABC9A12E4155640075755699 /* IndicatorDataSource.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = IndicatorDataSource.swift; sourceTree = ""; }; ABC9A1360FE305343B1049CF /* SwapRevokeConfirmationViewController.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = SwapRevokeConfirmationViewController.swift; sourceTree = ""; }; @@ -3789,6 +3819,7 @@ ABC9A309A58148C40912B964 /* ContactBookSettingsService.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = ContactBookSettingsService.swift; sourceTree = ""; }; ABC9A30A8F78E9C9AEE861F1 /* NftAssetCellFactory.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = NftAssetCellFactory.swift; sourceTree = ""; }; ABC9A352F3EAA38107897CEF /* WalletTokenBalanceService.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = WalletTokenBalanceService.swift; sourceTree = ""; }; + ABC9A37065F4A8459C416F0A /* BackupAppModule.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = BackupAppModule.swift; sourceTree = ""; }; ABC9A37521CD6E2CC5BA4E68 /* MarketCardView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = MarketCardView.swift; sourceTree = ""; }; ABC9A3758FE2D56036DF27FF /* ChartIndicatorsRepository.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = ChartIndicatorsRepository.swift; sourceTree = ""; }; ABC9A38082BD2EBE1BC8E11E /* WalletTokenService.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = WalletTokenService.swift; sourceTree = ""; }; @@ -3842,9 +3873,11 @@ ABC9A76776AD840DBFAA1804 /* CoinIndicatorViewItemFactory.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = CoinIndicatorViewItemFactory.swift; sourceTree = ""; }; ABC9A776346AF62265896CA1 /* CellElement.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = CellElement.swift; sourceTree = ""; }; ABC9A791A47F4F1E71B51B3B /* TokenSelectView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = TokenSelectView.swift; sourceTree = ""; }; + ABC9A7B2F7354F12F78A2F30 /* BackupAppViewModel.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = BackupAppViewModel.swift; sourceTree = ""; }; ABC9A7C3BC5FC664BBF14C4F /* SendEip721ViewModel.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = SendEip721ViewModel.swift; sourceTree = ""; }; ABC9A7D665A025E95697C757 /* AccountRestoreWarningManager.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = AccountRestoreWarningManager.swift; sourceTree = ""; }; ABC9A7D6C9D12C1F1F3A1218 /* SendBitcoinViewController.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = SendBitcoinViewController.swift; sourceTree = ""; }; + ABC9A7FA830E64B8DCA1A69A /* InputTextRow.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = InputTextRow.swift; sourceTree = ""; }; ABC9A7FC41B9F98871246E0E /* ImageCell.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = ImageCell.swift; sourceTree = ""; }; ABC9A80143F95E28346C81FE /* SendMemoInputService.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = SendMemoInputService.swift; sourceTree = ""; }; ABC9A806FD17A129212E3F7C /* NftAssetOverviewViewController.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = NftAssetOverviewViewController.swift; sourceTree = ""; }; @@ -3853,6 +3886,7 @@ ABC9A830FE79DBF62FD63CC4 /* ThemeMode.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = ThemeMode.swift; sourceTree = ""; }; ABC9A845B2969166028BA5F0 /* WalletConnectAppShowView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = WalletConnectAppShowView.swift; sourceTree = ""; }; ABC9A86EA911DA12C7A6AC20 /* WalletTokenBalanceViewItemFactory.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = WalletTokenBalanceViewItemFactory.swift; sourceTree = ""; }; + ABC9A86FE7E7F2DA93A5CB00 /* BackupNameView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = BackupNameView.swift; sourceTree = ""; }; ABC9A88E126AB21F856522A7 /* IntegerAmountInputView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = IntegerAmountInputView.swift; sourceTree = ""; }; ABC9A896A83640B618328FE1 /* EnsAddressParserItem.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = EnsAddressParserItem.swift; sourceTree = ""; }; ABC9A89726499CDB4F697EDD /* CipherParams.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = CipherParams.swift; sourceTree = ""; }; @@ -3877,6 +3911,7 @@ ABC9A9E0190FAD212E2E007F /* RestoreCloudPassphraseModule.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = RestoreCloudPassphraseModule.swift; sourceTree = ""; }; ABC9A9E2C039C005650491D2 /* WalletConnectAppShowModule.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = WalletConnectAppShowModule.swift; sourceTree = ""; }; ABC9A9F6635146BEBFB432D1 /* ChartCell.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = ChartCell.swift; sourceTree = ""; }; + ABC9AA0021E969A73DA7E177 /* BackupListView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = BackupListView.swift; sourceTree = ""; }; ABC9AA2491ADC4E5E089CD42 /* MetadataMonitor.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = MetadataMonitor.swift; sourceTree = ""; }; ABC9AA31438063F7AB7BDDC8 /* WalletConnectRequestMapper.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = WalletConnectRequestMapper.swift; sourceTree = ""; }; ABC9AA3B8927F9F138ABCFB8 /* WalletConnectAppShowService.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = WalletConnectAppShowService.swift; sourceTree = ""; }; @@ -3908,6 +3943,7 @@ ABC9AB9077A6A0ABE4909B76 /* IntegerFormAmountInputView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = IntegerFormAmountInputView.swift; sourceTree = ""; }; ABC9ABE473B354836327B3AC /* NftAssetOverviewService.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = NftAssetOverviewService.swift; sourceTree = ""; }; ABC9ABE97578DC667CBDC11A /* BaseSendViewController.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = BaseSendViewController.swift; sourceTree = ""; }; + ABC9ABF9E55990704693487F /* BackupDisclaimerView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = BackupDisclaimerView.swift; sourceTree = ""; }; ABC9ABFE62D22F9FB0B3409A /* DonateDescriptionDataSource.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = DonateDescriptionDataSource.swift; sourceTree = ""; }; ABC9AC09A586D88BAB3B9C67 /* WalletConnectListModule.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = WalletConnectListModule.swift; sourceTree = ""; }; ABC9AC0B5943DF3B61B20BF6 /* WalletConnectSessionManager.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = WalletConnectSessionManager.swift; sourceTree = ""; }; @@ -3917,9 +3953,11 @@ ABC9AC6A0B950C0AABD5A93E /* SendEip1155Service.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = SendEip1155Service.swift; sourceTree = ""; }; ABC9AC7983E4A81E421AB639 /* MarketCardTitleView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = MarketCardTitleView.swift; sourceTree = ""; }; ABC9AC8CCF3B57FDFC817356 /* ContactBookAddressService.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = ContactBookAddressService.swift; sourceTree = ""; }; + ABC9AC9C3F666BC8D15C7F53 /* BackupTypeView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = BackupTypeView.swift; sourceTree = ""; }; ABC9ACE2CCBDF21572F5600C /* ChartIndicatorSettingsViewModel.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = ChartIndicatorSettingsViewModel.swift; sourceTree = ""; }; ABC9ACE7CB7CC9C118C72559 /* SendEip721Service.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = SendEip721Service.swift; sourceTree = ""; }; ABC9ACE88105815BFC477D71 /* WalletConnectPairingViewController.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = WalletConnectPairingViewController.swift; sourceTree = ""; }; + ABC9ACEC3169A9F01B55921A /* InputTextView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = InputTextView.swift; sourceTree = ""; }; ABC9ACF1ACFDFD53E2502C30 /* SendFeeViewModel.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = SendFeeViewModel.swift; sourceTree = ""; }; ABC9ACF1F55164BDFD049793 /* SendBitcoinAmountInputService.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = SendBitcoinAmountInputService.swift; sourceTree = ""; }; ABC9ACF418357FF7AFC64B3F /* UniswapV3TradeService.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = UniswapV3TradeService.swift; sourceTree = ""; }; @@ -3932,6 +3970,7 @@ ABC9AD3F677671FB57CCD886 /* WalletConnectPendingRequestsService.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = WalletConnectPendingRequestsService.swift; sourceTree = ""; }; ABC9AD448DC071D8800C6B12 /* WalletTokenBalanceCustomAmountCell.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = WalletTokenBalanceCustomAmountCell.swift; sourceTree = ""; }; ABC9AD5CB1911A698718213F /* BackupCryptoHelper.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = BackupCryptoHelper.swift; sourceTree = ""; }; + ABC9AD8AEA1E3B1198DDECCE /* BackupManagerModule.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = BackupManagerModule.swift; sourceTree = ""; }; ABC9ADB77831DCB474B24C8A /* SendFeeService.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = SendFeeService.swift; sourceTree = ""; }; ABC9ADC1A3B17225B6CC0869 /* MarketCategoryMarketCapFetcher.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = MarketCategoryMarketCapFetcher.swift; sourceTree = ""; }; ABC9ADE822BC024F9B798211 /* BottomGradientHolder.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = BottomGradientHolder.swift; sourceTree = ""; }; @@ -3939,6 +3978,7 @@ ABC9ADFD9DA59BD2FB21C51B /* MarketDiscoveryCategoryService.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = MarketDiscoveryCategoryService.swift; sourceTree = ""; }; ABC9AE12A5E8B9FB24FFE42F /* ContactBookHelper.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = ContactBookHelper.swift; sourceTree = ""; }; ABC9AE15C187118DE6F0CE7B /* WalletConnectPendingRequestsViewModel.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = WalletConnectPendingRequestsViewModel.swift; sourceTree = ""; }; + ABC9AE522F09C5E7029CA86E /* CheckboxStyle.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = CheckboxStyle.swift; sourceTree = ""; }; ABC9AE5FD79ECC4AC85B86FA /* WalletConnectListViewController.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = WalletConnectListViewController.swift; sourceTree = ""; }; ABC9AE62C0399849EFB5C158 /* WalletConnectPairingViewModel.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = WalletConnectPairingViewModel.swift; sourceTree = ""; }; ABC9AE6D2CD14194802E7976 /* SendZcashService.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = SendZcashService.swift; sourceTree = ""; }; @@ -3948,11 +3988,13 @@ ABC9AEA1D717D8CED8462AB0 /* WalletConnectMainViewModel.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = WalletConnectMainViewModel.swift; sourceTree = ""; }; ABC9AEAD18F73D4FBE05783D /* Contact.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = Contact.swift; sourceTree = ""; }; ABC9AEC034DE5784F55BD5F3 /* PseudoAccessoryView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = PseudoAccessoryView.swift; sourceTree = ""; }; + ABC9AEC2C4752498AF7A7A2E /* BackupPasswordView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = BackupPasswordView.swift; sourceTree = ""; }; ABC9AECEEB35D57CB0965E79 /* WalletBackup.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = WalletBackup.swift; sourceTree = ""; }; ABC9AF12879C62002DFE946A /* RestoreCloudPassphraseViewController.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = RestoreCloudPassphraseViewController.swift; sourceTree = ""; }; ABC9AF15BD67548E6D755CA0 /* SendBinanceViewController.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = SendBinanceViewController.swift; sourceTree = ""; }; ABC9AF26FDCB363793BF66E1 /* Integer.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = Integer.swift; sourceTree = ""; }; ABC9AF2B063727B7EABFD9A3 /* RsiIndicatorDataSource.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = RsiIndicatorDataSource.swift; sourceTree = ""; }; + ABC9AF395EA01B43D6D77C43 /* ActivityViewController.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = ActivityViewController.swift; sourceTree = ""; }; ABC9AF6AA02DA39787C053F0 /* BackupCloudPassphraseService.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = BackupCloudPassphraseService.swift; sourceTree = ""; }; ABC9AF8E8DE67732371A00E0 /* FeePriceScale.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = FeePriceScale.swift; sourceTree = ""; }; ABC9AF9C0D0174A5B6A91F13 /* NftAssetViewController.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = NftAssetViewController.swift; sourceTree = ""; }; @@ -4641,6 +4683,10 @@ 11B3584D2C3754A605975D6C /* SecondaryCircleButtonStyle.swift */, 11B351FDDBEF227E161F6A0E /* PageDescription.swift */, 11B356EF92FFD23F4385A991 /* ListStyle.swift */, + ABC9AE522F09C5E7029CA86E /* CheckboxStyle.swift */, + ABC9ACEC3169A9F01B55921A /* InputTextView.swift */, + ABC9A7FA830E64B8DCA1A69A /* InputTextRow.swift */, + ABC9A104D916039D690E454E /* Shake.swift */, ); path = SwiftUI; sourceTree = ""; @@ -4731,6 +4777,7 @@ 11B352972B14FA6EBEFD6904 /* Text.swift */, 11B352648C452D611F1EDF61 /* Image.swift */, 11B352978EC570F59F442BD5 /* View.swift */, + ABC9AF395EA01B43D6D77C43 /* ActivityViewController.swift */, ); path = Extensions; sourceTree = ""; @@ -5380,6 +5427,7 @@ 11B35C3CD4FFDE56E8E30B80 /* BlockchainSettings */, D06F756D2A8BA33000184227 /* Donate */, 11B35710326AFD7334D8D044 /* SimpleActivate */, + ABC9A9C9D65EAD890AF617A4 /* BackupApp */, ); path = Settings; sourceTree = ""; @@ -7026,6 +7074,14 @@ path = DataSources; sourceTree = ""; }; + ABC9A3BF4876DB959C5E4DE2 /* BackupDisclaimer */ = { + isa = PBXGroup; + children = ( + ABC9ABF9E55990704693487F /* BackupDisclaimerView.swift */, + ); + path = BackupDisclaimer; + sourceTree = ""; + }; ABC9A3CED3BD03C1DBF797E2 /* Passphrase */ = { isa = PBXGroup; children = ( @@ -7089,6 +7145,14 @@ path = Send; sourceTree = ""; }; + ABC9A5A9AA93B6487D21EFF9 /* BackupList */ = { + isa = PBXGroup; + children = ( + ABC9AA0021E969A73DA7E177 /* BackupListView.swift */, + ); + path = BackupList; + sourceTree = ""; + }; ABC9A5F2A0999FCFDDEF1BA3 /* AmountCaution */ = { isa = PBXGroup; children = ( @@ -7202,6 +7266,14 @@ path = Components; sourceTree = ""; }; + ABC9A77088096A37E5C42191 /* BackupType */ = { + isa = PBXGroup; + children = ( + ABC9AC9C3F666BC8D15C7F53 /* BackupTypeView.swift */, + ); + path = BackupType; + sourceTree = ""; + }; ABC9A7E9EAE24647C0700B39 /* RestoreCloud */ = { isa = PBXGroup; children = ( @@ -7225,6 +7297,21 @@ path = V2; sourceTree = ""; }; + ABC9A9C9D65EAD890AF617A4 /* BackupApp */ = { + isa = PBXGroup; + children = ( + ABC9AD5DC41717F92AC2151C /* BackupManager */, + ABC9A77088096A37E5C42191 /* BackupType */, + ABC9A5A9AA93B6487D21EFF9 /* BackupList */, + ABC9A3BF4876DB959C5E4DE2 /* BackupDisclaimer */, + ABC9AADD1C32448E8AA5D25F /* BackupName */, + ABC9A7B2F7354F12F78A2F30 /* BackupAppViewModel.swift */, + ABC9A37065F4A8459C416F0A /* BackupAppModule.swift */, + ABC9ADC8D7C3B51B96589BC0 /* BackupPassword */, + ); + path = BackupApp; + sourceTree = ""; + }; ABC9AA15272B5421D314CDD9 /* AmountInput */ = { isa = PBXGroup; children = ( @@ -7269,6 +7356,14 @@ path = Bitcoin; sourceTree = ""; }; + ABC9AADD1C32448E8AA5D25F /* BackupName */ = { + isa = PBXGroup; + children = ( + ABC9A86FE7E7F2DA93A5CB00 /* BackupNameView.swift */, + ); + path = BackupName; + sourceTree = ""; + }; ABC9AB313E49F27BBB0C9AED /* DataSources */ = { isa = PBXGroup; children = ( @@ -7334,6 +7429,15 @@ path = Platforms; sourceTree = ""; }; + ABC9AD5DC41717F92AC2151C /* BackupManager */ = { + isa = PBXGroup; + children = ( + ABC9A121EDBB1FF651D514D1 /* BackupManagerView.swift */, + ABC9AD8AEA1E3B1198DDECCE /* BackupManagerModule.swift */, + ); + path = BackupManager; + sourceTree = ""; + }; ABC9ADC8373B45DF33E35DCA /* Binance */ = { isa = PBXGroup; children = ( @@ -7345,6 +7449,14 @@ path = Binance; sourceTree = ""; }; + ABC9ADC8D7C3B51B96589BC0 /* BackupPassword */ = { + isa = PBXGroup; + children = ( + ABC9AEC2C4752498AF7A7A2E /* BackupPasswordView.swift */, + ); + path = BackupPassword; + sourceTree = ""; + }; ABC9ADCD89D02064F90038F5 /* ContactBookContact */ = { isa = PBXGroup; children = ( @@ -9301,6 +9413,20 @@ 11B356A35A5981DD231E580C /* ListStyle.swift in Sources */, 11B357C425D633543FD109C3 /* DuressModeSelectView.swift in Sources */, 11B358D35D2270FD78C6EF82 /* AutoLockPeriod.swift in Sources */, + ABC9A50F1811EEABE61913F7 /* BackupManagerView.swift in Sources */, + ABC9A81D09F820C68D4F75FB /* BackupManagerModule.swift in Sources */, + ABC9A5125874E09194BA8532 /* BackupTypeView.swift in Sources */, + ABC9AF95141EA649524FBF88 /* CheckboxStyle.swift in Sources */, + ABC9A6C05017D9267567AB73 /* BackupDisclaimerView.swift in Sources */, + ABC9A7244B2D23BDA40C407C /* BackupNameView.swift in Sources */, + ABC9A20B7F90F5E741A74823 /* BackupAppViewModel.swift in Sources */, + ABC9A79CFCEBAC442A1B791D /* BackupAppModule.swift in Sources */, + ABC9ACE31EC94ABC5B325693 /* BackupListView.swift in Sources */, + ABC9A09E0B614E5B4E32B7F9 /* InputTextView.swift in Sources */, + ABC9ABE3F52BF2307533D8FB /* InputTextRow.swift in Sources */, + ABC9A36D3A4EEABF6EA6DBA0 /* Shake.swift in Sources */, + ABC9A06CA0D364D4E1261C87 /* BackupPasswordView.swift in Sources */, + ABC9A4A21CFBA188A7EEC930 /* ActivityViewController.swift in Sources */, ); runOnlyForDeploymentPostprocessing = 0; }; @@ -10613,6 +10739,20 @@ 11B353577381981235B90A82 /* ListStyle.swift in Sources */, 11B354DC983042AD922339A6 /* DuressModeSelectView.swift in Sources */, 11B3511DAD3881FDE2419A64 /* AutoLockPeriod.swift in Sources */, + ABC9A8CAD0A3AAB7398C2A30 /* BackupManagerView.swift in Sources */, + ABC9A77A41517173B610E0FC /* BackupManagerModule.swift in Sources */, + ABC9AB65B10FDB5F30E2731D /* BackupTypeView.swift in Sources */, + ABC9A2035980B70E1C0790A8 /* CheckboxStyle.swift in Sources */, + ABC9A4EE80A57455DB2CCD4F /* BackupDisclaimerView.swift in Sources */, + ABC9A94A82E55F437BD3FE68 /* BackupNameView.swift in Sources */, + ABC9A13B7F43722709570161 /* BackupAppViewModel.swift in Sources */, + ABC9A99A45187C36D48840F8 /* BackupAppModule.swift in Sources */, + ABC9AA05527830EFFCFB98E3 /* BackupListView.swift in Sources */, + ABC9AE51262C09EABF5CCEEE /* InputTextView.swift in Sources */, + ABC9A542CA987F09C93F04A9 /* InputTextRow.swift in Sources */, + ABC9A7C2087C3A641C3F9AD4 /* Shake.swift in Sources */, + ABC9A54E6E15E96271525339 /* BackupPasswordView.swift in Sources */, + ABC9A12A4D114A2E4F4C711A /* ActivityViewController.swift in Sources */, ); runOnlyForDeploymentPostprocessing = 0; }; diff --git a/UnstoppableWallet/UnstoppableWallet/Assets.xcassets/Icons/file_24.imageset/Contents.json b/UnstoppableWallet/UnstoppableWallet/Assets.xcassets/Icons/file_24.imageset/Contents.json new file mode 100644 index 0000000000..3d16665f31 --- /dev/null +++ b/UnstoppableWallet/UnstoppableWallet/Assets.xcassets/Icons/file_24.imageset/Contents.json @@ -0,0 +1,22 @@ +{ + "images" : [ + { + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "file@2x.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "filename" : "file@3x.png", + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/UnstoppableWallet/UnstoppableWallet/Assets.xcassets/Icons/file_24.imageset/file@2x.png b/UnstoppableWallet/UnstoppableWallet/Assets.xcassets/Icons/file_24.imageset/file@2x.png new file mode 100644 index 0000000000000000000000000000000000000000..8cf99e8a82e559e9e53b2ea82dd1e68060e1587f GIT binary patch literal 472 zcmeAS@N?(olHy`uVBq!ia0vp^1|ZDA1|-9oezpTC&H|6fVg?2=RS;(M3{v?36l5$8 za(7}_cTVOdki(Mh=P`1UZfiI}Z`@uPtN0o1!>y|`x?g_NHnSU&BPoPQFrFCJOU78d)4yi8A zIksueJ;`LPn!O8?*~2>Zx}Hy(;hE!>>=At6$!){1T{`KkhLdlcRa?qv(b~~6Bhw|r zwpK-Ljp2f?vv1bahd$x_XYKgewNZ~FYpKYi7f*}hmme%=ND|L5ve_@JIgHAO}>A)G^=m%%$oOxHLg7+O3#jP{MJ0u%hAyrE@#0!$1+jJ`oS5_ zdv@0^?;33nIu9CKqi0zuD=SQh9BMI~BDtW96(Z{N=Fb??u*zwfLc z2yEfrc1y2AH>Xy%T=8>r|7}y}KE~oZz8XsMW{Z>A|E+4sVqz~1SGaLdiy^vWQroF> zE-t;9Y3t;x*PkqyEpwK69_QRCYJAfRo~B;aTcN4+BHPF4idaXNOf^mN`9fTID9hvet3??amVmTGzNe`?rXtSo{CK zMXzeUKiq!%zeeBsyc09R$~H$YYv&KJeXb_s`d3eKF>8poMgRLWgC$~;kDZg=PSH&c=`njAxHWXl>kc@12?Gl^YyM?^KE| z`Hh1E7t0bbp%Hy_SJCafytm5~;tj7~ zK4p11UCwdS6d&iv=#w#@f4!MLqvdn1?rP>|3aM8`7&QeZEZQh~bl1}E)K+bMw+T!j z5q!icwvgBNj#_G#o{Mrg*>gO6>(M;#WzVJl)bBN|IK5qJZG~I&o`>N--5ykQ_9VY+ z{}+><`}H!@sg0NSnXf6BUVGVXlhLf5vsI4El%#)BXDTjQRvK}^>c~d}&comMC3EUu zIM1A8_+CD9@0A7{&X-0!|Nl*yaOS+_&mG0jntN?*i&>u>{rI)DM`h;brV01fZ)~|d zapLC5-#K5Y{0-X0c<28K*Hg Data { let backup = try appBackupProvider.fullBackup( fields: fields, passphrase: passphrase ) + return try JSONEncoder().encode(backup) + } + + func file(fields: [AppBackupProvider.Field], passphrase: String, name: String) throws -> URL { + let data = try data(fields: fields, passphrase: passphrase) + + // save book to temporary file + guard let temporaryFileUrl = ContactBookManager.localUrl?.appendingPathComponent(name + ".json") else { + throw FileStorage.StorageError.cantCreateFile + } + try data.write(to: temporaryFileUrl) + return temporaryFileUrl + } + + func save(fields: [AppBackupProvider.Field], passphrase: String, name: String) throws { do { - let encoder = JSONEncoder() - let data = try encoder.encode(backup) - let encoded = try JSONEncoder().encode(backup) + let encoded = try data(fields: fields, passphrase: passphrase) try save(encoded: encoded, name: name) } catch { logger?.log(level: .debug, message: "CloudAccountManager.downloadItems, can't save \(name). Because: \(error)") diff --git a/UnstoppableWallet/UnstoppableWallet/Core/Managers/EvmSyncSourceManager.swift b/UnstoppableWallet/UnstoppableWallet/Core/Managers/EvmSyncSourceManager.swift index e4561c32a1..953b4e6283 100644 --- a/UnstoppableWallet/UnstoppableWallet/Core/Managers/EvmSyncSourceManager.swift +++ b/UnstoppableWallet/UnstoppableWallet/Core/Managers/EvmSyncSourceManager.swift @@ -189,11 +189,17 @@ extension EvmSyncSourceManager { } } - func customSyncSources(blockchainType: BlockchainType) -> [EvmSyncSource] { + func customSyncSources(blockchainType: BlockchainType?) -> [EvmSyncSource] { do { - let records = try evmSyncSourceStorage.records(blockchainTypeUid: blockchainType.uid) + let records: [EvmSyncSourceRecord] + if let blockchainType { + records = try evmSyncSourceStorage.records(blockchainTypeUid: blockchainType.uid) + } else { + records = try evmSyncSourceStorage.getAll() + } return records.compactMap { record in + let blockchainType = BlockchainType(uid: record.blockchainTypeUid) guard let url = URL(string: record.url), let scheme = url.scheme else { return nil } diff --git a/UnstoppableWallet/UnstoppableWallet/Modules/Backup/ICloud/Passphrase/BackupCloudPassphraseViewModel.swift b/UnstoppableWallet/UnstoppableWallet/Modules/Backup/ICloud/Passphrase/BackupCloudPassphraseViewModel.swift index 53331cd609..076f1b8988 100644 --- a/UnstoppableWallet/UnstoppableWallet/Modules/Backup/ICloud/Passphrase/BackupCloudPassphraseViewModel.swift +++ b/UnstoppableWallet/UnstoppableWallet/Modules/Backup/ICloud/Passphrase/BackupCloudPassphraseViewModel.swift @@ -83,15 +83,11 @@ extension BackupCloudPassphraseViewModel { } catch { switch error { case BackupCrypto.ValidationError.emptyPassphrase: - passphraseCaution = Caution(text: "backup.cloud.password.error.empty_passphrase".localized, type: .error) + passphraseCaution = Caution(text: error.localizedDescription, type: .error) case BackupCrypto.ValidationError.simplePassword: - passphraseCaution = Caution(text: "backup.cloud.password.error.minimum_requirement".localized, type: .error) + passphraseCaution = Caution(text: error.localizedDescription, type: .error) case BackupCloudPassphraseService.CreateError.invalidConfirmation: passphraseConfirmationCaution = Caution(text: "backup.cloud.password.confirm.error.doesnt_match".localized, type: .error) - case BackupCloudPassphraseService.CreateError.urlNotAvailable: - showErrorSubject.send("backup.cloud.not_available".localized) - case BackupCloudPassphraseService.CreateError.cantSaveFile: - showErrorSubject.send("backup.cloud.cant_create_file".localized) default: showErrorSubject.send(error.smartDescription) } @@ -100,3 +96,32 @@ extension BackupCloudPassphraseViewModel { } } + +extension BackupCrypto.ValidationError: LocalizedError { + public var errorDescription: String? { + switch self { + case .emptyPassphrase: return "backup.cloud.password.error.empty_passphrase".localized + case .simplePassword: return "backup.cloud.password.error.minimum_requirement".localized + } + } +} + +extension BackupCloudPassphraseService.CreateError: LocalizedError { + public var errorDescription: String? { + switch self { + case .urlNotAvailable: return "backup.cloud.not_available".localized + case .cantSaveFile: return "backup.cloud.cant_create_file".localized + case .invalidConfirmation: return "invalid confirmation".localized + } + } +} + +extension CloudBackupManager.BackupError: LocalizedError { + public var errorDescription: String? { + switch self { + case .urlNotAvailable: return "backup.cloud.not_available".localized + case .itemNotFound: return nil + } + } +} + diff --git a/UnstoppableWallet/UnstoppableWallet/Modules/BottomSheet/BottomSheetModule.swift b/UnstoppableWallet/UnstoppableWallet/Modules/BottomSheet/BottomSheetModule.swift index ce38788fd5..b6ae2ef412 100644 --- a/UnstoppableWallet/UnstoppableWallet/Modules/BottomSheet/BottomSheetModule.swift +++ b/UnstoppableWallet/UnstoppableWallet/Modules/BottomSheet/BottomSheetModule.swift @@ -1,4 +1,5 @@ import UIKit +import SwiftUI import ThemeKit import ComponentKit import SectionsTableView @@ -146,3 +147,20 @@ extension BottomSheetModule { } } + +struct ViewWrapper: UIViewControllerRepresentable { + typealias UIViewControllerType = UIViewController + + let viewController: UIViewController + + init(_ viewController: UIViewController) { + self.viewController = viewController + } + + func makeUIViewController(context _: Context) -> UIViewController { + viewController + } + + func updateUIViewController(_: UIViewController, context _: Context) {} +} + diff --git a/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupAppModule.swift b/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupAppModule.swift new file mode 100644 index 0000000000..1ed8a0a95b --- /dev/null +++ b/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupAppModule.swift @@ -0,0 +1,52 @@ +import SwiftUI + +struct BackupAppModule { + static func view(backupPresented: Binding) -> some View { + let viewModel = BackupAppViewModel( + accountManager: App.shared.accountManager, + contactManager: App.shared.contactManager, + cloudBackupManager: App.shared.cloudBackupManager, + favoritesManager: App.shared.favoritesManager, + evmSyncSourceManager: App.shared.evmSyncSourceManager + ) + + return BackupTypeView(viewModel: viewModel, backupPresented: backupPresented) + } +} + +extension BackupAppModule { + enum Destination: String, CaseIterable, Identifiable { + case cloud + case local + + var id: Self { + self + } + + var backupDisclaimer: BackupDestinationDisclaimer { + switch self { + case .cloud: + return BackupDestinationDisclaimer( + title: "backup.disclaimer.cloud.title".localized, + highlightedDescription: "backup.disclaimer.cloud.description".localized, + selectedCheckboxText: "backup.disclaimer.cloud.checkbox_label".localized, + buttonTitle: "button.next".localized + ) + case .local: + return BackupDestinationDisclaimer( + title: "backup.disclaimer.file.title".localized, + highlightedDescription: "backup.disclaimer.file.description".localized, + selectedCheckboxText: "backup.disclaimer.file.checkbox_label".localized, + buttonTitle: "button.next".localized + ) + } + } + } + + struct BackupDestinationDisclaimer { + let title: String + let highlightedDescription: String + let selectedCheckboxText: String + let buttonTitle: String + } +} diff --git a/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupAppViewModel.swift b/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupAppViewModel.swift new file mode 100644 index 0000000000..77dc46cf5b --- /dev/null +++ b/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupAppViewModel.swift @@ -0,0 +1,379 @@ +import Combine +import ComponentKit +import Foundation + +class BackupAppViewModel: ObservableObject { + static let backupNamePrefix = "App Backup" + let accountManager: AccountManager + let contactManager: ContactBookManager + let cloudBackupManager: CloudBackupManager + let favoritesManager: FavoritesManager + let evmSyncSourceManager: EvmSyncSourceManager + + private var cancellables = Set() + + // Type ViewModel + @Published var cloudAvailable: Bool + @Published var destination: BackupAppModule.Destination? { + didSet { + // need to reset future fields: + name = nextName + password = AppConfig.defaultPassphrase + confirm = AppConfig.defaultPassphrase + + accountItems = accounts(watch: false) + .map { item(account: $0) } + + otherItems = getOtherItems() + selected = accountIds.reduce(into: [:]) { $0[$1] = true } + } + } + + // Configuration ViewModel + @Published var selected: [String: Bool] = [:] + @Published var accountItems: [AccountItem] = [] + @Published var otherItems: [Item] = [] + @Published var disclaimerPushed = false { + didSet { + // need to reset future fields: + name = nextName + password = AppConfig.defaultPassphrase + confirm = AppConfig.defaultPassphrase + } + } + + // Disclaimer ViewModel + @Published var namePushed = false { + didSet { + // need to reset future fields: + name = nextName + password = AppConfig.defaultPassphrase + confirm = AppConfig.defaultPassphrase + } + } + + // Name ViewModel + @Published var nameCautionState: CautionState = .none + @Published var name: String = "" { + didSet { + validateName() + } + } + @Published var passwordPushed = false { + didSet { + // need to reset future fields: + password = AppConfig.defaultPassphrase + confirm = AppConfig.defaultPassphrase + } + } + + // Password ViewModel + @Published var passwordCautionState: CautionState = .none + @Published var password: String = AppConfig.defaultPassphrase { + didSet { + validatePasswords() + } + } + + @Published var confirmCautionState: CautionState = .none + @Published var confirm: String = AppConfig.defaultPassphrase { + didSet { + validatePasswords() + } + } + + @Published var passwordButtonDisabled = true + @Published var passwordButtonProcessing = false + + @Published var sharePresented: URL? + + init(accountManager: AccountManager, contactManager: ContactBookManager, cloudBackupManager: CloudBackupManager, favoritesManager: FavoritesManager, evmSyncSourceManager: EvmSyncSourceManager) { + self.accountManager = accountManager + self.contactManager = contactManager + self.cloudBackupManager = cloudBackupManager + self.favoritesManager = favoritesManager + self.evmSyncSourceManager = evmSyncSourceManager + + cloudAvailable = cloudBackupManager.iCloudUrl != nil + cloudBackupManager.$state + .sink(receiveValue: { [weak self] state in + switch state { + case .error: self?.cloudAvailable = false + default: self?.cloudAvailable = true + } + }) + .store(in: &cancellables) + + accountItems = accounts(watch: false) + .map { item(account: $0) } + + otherItems = getOtherItems() + selected = accountIds.reduce(into: [:]) { $0[$1] = true } + name = nextName + + validatePasswords() + } +} + +// Account Page ViewModel +extension BackupAppViewModel { + private func accounts(watch: Bool) -> [Account] { + accountManager + .accounts + .filter { $0.watchAccount == watch } + } + + private var accountIds: [String] { + accounts(watch: false) + .map { $0.id } + } + + private func item(account: Account) -> AccountItem { + var alertSubtitle: String? + let hasAlertDescription = !(account.backedUp || cloudBackupManager.backedUp(uniqueId: account.type.uniqueId())) + if account.nonStandard { + alertSubtitle = "manage_accounts.migration_required".localized + } else if hasAlertDescription { + alertSubtitle = "manage_accounts.backup_required".localized + } + + let showAlert = alertSubtitle != nil || account.nonRecommended + + let cautionType: CautionType? = showAlert ? .error : .none + let description = alertSubtitle ?? account.type.detailedDescription + + return AccountItem( + accountId: account.id, + name: account.name, + description: description, + cautionType: cautionType + ) + } + + private func getOtherItems() -> [Item] { + var items = [Item]() + + let watchAccountCount = accounts(watch: true).count + if watchAccountCount != 0 { + items.append(Item( + title: "backup_list.other.watch_account.title".localized, + description: "backup_list.other.watch_account.description".localized(watchAccountCount) + )) + } + + let watchlistCount = favoritesManager.allCoinUids.count + if watchlistCount != 0 { + items.append(Item( + title: "backup_list.other.watchlist.title".localized, + description: "backup_list.other.watchlist.description".localized(watchlistCount) + )) + } + + let contacts = contactManager.all ?? [] + let contactAddressCount = contacts.reduce(into: 0) { $0 += $1.addresses.count } + if contactAddressCount != 0 { + items.append(Item( + title: "backup_list.other.contacts.title".localized, + description: "backup_list.other.contacts.description".localized(contactAddressCount) + )) + } + + let blockchainSourcesCount = evmSyncSourceManager.customSyncSources(blockchainType: nil).count + if blockchainSourcesCount != 0 { + items.append(Item( + title: "backup_list.other.blockchain_settings.title".localized, + description: "backup_list.other.blockchain_settings.description".localized(blockchainSourcesCount) + )) + } + items.append(Item( + title: "backup_list.other.app_settings.title".localized, + description: "backup_list.other.app_settings.description".localized + )) + + return items + } + + var configuration: [AppBackupProvider.Field] { + var fields = [AppBackupProvider.Field.settings] + + var accountIds = accounts(watch: true).map { $0.id } + selected.forEach { id, selected in + if selected { + accountIds.append(id) + } + } + + fields.append(.accounts(ids: accountIds)) + + let contacts = contactManager.all ?? [] + if contacts.count != 0 { + fields.append(.contacts) + } + + if !favoritesManager.allCoinUids.isEmpty { + fields.append(.watchlist) + } + + return fields + } +} + +extension BackupAppViewModel { + func toggle(item: AccountItem) { + selected[item.accountId]?.toggle() + } +} + +// Backup Name VieeModel +extension BackupAppViewModel { + var nextName: String { + let name = { (_: String) in [Self.backupNamePrefix, "1"].joined(separator: " ") } + switch destination { + case .cloud: + let exists = cloudBackupManager + .existFilenames + .filter { $0.hasPrefix(Self.backupNamePrefix) } + .sorted() + for i in 1 ..< exists.count + 1 { + let newName = name(i.description) + if !exists.contains(where: { $0.lowercased() == newName.lowercased() }) { + return newName + } + } + return name((exists.count + 1).description) + default: + return name("1") + } + } + + func validateName() { + if name.isEmpty { + nameCautionState = .caution(.init(text: NameError.empty.localizedDescription, type: .error)) + } else if (cloudBackupManager.existFilenames + [Self.backupNamePrefix + "2"]).contains(where: { $0.lowercased() == name.lowercased() }) { + nameCautionState = .caution(.init(text: NameError.alreadyExist.localizedDescription, type: .error)) + } else { + nameCautionState = .none + } + } + + func validatePasswords() { + var buttonDisabled = false + if password.isEmpty { + buttonDisabled = true + confirmCautionState = .none + } else { + do { + try BackupCrypto.validate(passphrase: password) + passwordCautionState = .none + } catch { + passwordCautionState = .caution(.init(text: error.localizedDescription, type: .error)) + buttonDisabled = true + } + } + + if confirm.isEmpty { + buttonDisabled = true + confirmCautionState = .none + } else { + do { + try BackupCrypto.validate(passphrase: confirm) + if password != confirm { + buttonDisabled = true + confirmCautionState = .caution( + .init( + text: "backup.cloud.password.confirm.error.doesnt_match".localized, + type: .error + ) + ) + } else { + confirmCautionState = .none + } + } catch { + confirmCautionState = .caution(.init(text: error.localizedDescription, type: .error)) + buttonDisabled = true + } + } + + passwordButtonDisabled = buttonDisabled + } + + @MainActor + private func showSuccess() { + HudHelper.instance.show(banner: .savedToCloud) + } + + @MainActor + private func show(error: Error) { + HudHelper.instance.show(banner: .error(string: error.localizedDescription)) + } + + func onTapSave() { + passwordButtonProcessing = true + + Task { + switch destination { + case .none: () + case .cloud: + do { + try cloudBackupManager.save(fields: configuration, passphrase: password, name: name) + passwordButtonProcessing = false + await showSuccess() + } catch { + passwordButtonProcessing = false + await show(error: error) + } + case .local: + do { + let url = try cloudBackupManager.file(fields: configuration, passphrase: password, name: name) + sharePresented = url + passwordButtonProcessing = false + } catch { + passwordButtonProcessing = false + await show(error: error) + } + } + } + } +} + +extension BackupAppViewModel { + struct AccountItem: Comparable, Identifiable { + let accountId: String + let name: String + let description: String + let cautionType: CautionType? + + static func < (lhs: AccountItem, rhs: AccountItem) -> Bool { + lhs.name < rhs.name + } + + static func == (lhs: AccountItem, rhs: AccountItem) -> Bool { + lhs.accountId == rhs.accountId + } + + var id: String { + accountId + } + } + + struct Item: Identifiable { + let title: String + let description: String + + var id: String { + title + } + } + + enum NameError: Error, LocalizedError { + case empty + case alreadyExist + + var errorDescription: String? { + switch self { + case .empty: return "backup.cloud.name.error.empty".localized + case .alreadyExist: return "backup.cloud.name.error.already_exist".localized + } + } + } +} diff --git a/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupDisclaimer/BackupDisclaimerView.swift b/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupDisclaimer/BackupDisclaimerView.swift new file mode 100644 index 0000000000..73125c8f6c --- /dev/null +++ b/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupDisclaimer/BackupDisclaimerView.swift @@ -0,0 +1,53 @@ +import SDWebImageSwiftUI +import SwiftUI +import ThemeKit + +struct BackupDisclaimerView: View { + @ObservedObject var viewModel: BackupAppViewModel + @Binding var backupPresented: Bool + + @State var isOn: Bool = true + + var body: some View { + let backupDisclaimer = (viewModel.destination ?? .local).backupDisclaimer + + ThemeView { + BottomGradientWrapper { + VStack(spacing: .margin32) { + HighlightedTextView(text: backupDisclaimer.highlightedDescription, style: .warning) + ListSection { + ClickableRow(action: { + isOn.toggle() + }) { + Toggle(isOn: $isOn) {} + .labelsHidden() + .toggleStyle(CheckboxStyle()) + + Text(backupDisclaimer.selectedCheckboxText).themeSubhead2(color: .themeLeah) + } + } + } + .padding(EdgeInsets(top: .margin12, leading: .margin16, bottom: .margin32, trailing: .margin16)) + } bottomContent: { + NavigationLink( + destination: BackupNameView(viewModel: viewModel, backupPresented: $backupPresented), + isActive: $viewModel.namePushed + ) { + Button(action: { viewModel.namePushed = true }) { + Text("button.next".localized) + } + .buttonStyle(PrimaryButtonStyle(style: .yellow)) + .disabled(!isOn) + } + } + } + .navigationBarTitle(backupDisclaimer.title) + .toolbar { + ToolbarItem(placement: .navigationBarTrailing) { + Button("button.cancel".localized) { + backupPresented = false + } + } + } + } +} diff --git a/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupList/BackupListView.swift b/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupList/BackupListView.swift new file mode 100644 index 0000000000..7c5f512838 --- /dev/null +++ b/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupList/BackupListView.swift @@ -0,0 +1,104 @@ +import SDWebImageSwiftUI +import SwiftUI +import ThemeKit + +struct BackupListView: View { + @ObservedObject var viewModel: BackupAppViewModel + @Binding var backupPresented: Bool + + var body: some View { + ThemeView { + BottomGradientWrapper { + VStack(spacing: .margin24) { + VStack(spacing: 0) { + ListSectionHeader(text: "backup_list.header.wallets".localized) + + ListSection { + ForEach(viewModel.accountItems, id: \.accountId) { (item: BackupAppViewModel.AccountItem) in + if viewModel.selected[item.id] != nil { + let selected = binding(for: item.accountId) + + ClickableRow(action: { + viewModel.toggle(item: item) + }) { + HStack { + AccountView(item: item) + + Toggle(isOn: selected) {} + .labelsHidden() + .toggleStyle(CheckboxStyle()) + } + } + } else { + ListRow { + AccountView(item: item) + } + } + } + } + } + + VStack(spacing: 0) { + ListSectionHeader(text: "backup_list.header.other".localized) + + ListSection { + ForEach(viewModel.otherItems) { (item: BackupAppViewModel.Item) in + ListRow { + VStack(spacing: 1) { + Text(item.title).themeBody() + + Text(item.description).themeSubhead2() + } + } + } + } + } + } + .padding(EdgeInsets(top: .margin12, leading: .margin16, bottom: .margin32, trailing: .margin16)) + } bottomContent: { + NavigationLink( + destination: BackupDisclaimerView(viewModel: viewModel, backupPresented: $backupPresented), + isActive: $viewModel.disclaimerPushed + ) { + Button(action: { + viewModel.disclaimerPushed = true + }) { + Text("button.next".localized) + } + .buttonStyle(PrimaryButtonStyle(style: .yellow)) + } + } + .navigationTitle("backup_list.title".localized) + .navigationBarTitleDisplayMode(.inline) + .toolbar { + Button("button.cancel".localized) { + backupPresented = false + } + } + } + } + + private func binding(for key: String) -> Binding { + .init( + get: { viewModel.selected[key, default: true] }, + set: { viewModel.selected[key] = $0 } + ) + } +} + +extension BackupListView { + struct AccountView: View { + var item: BackupAppViewModel.AccountItem + + var body: some View { + let color: Color? = item.cautionType.map { $0 == .error ? .themeLucian : .themeJacob } + + VStack(spacing: 1) { + Text(item.name).themeBody() + + Text(item.description) + .themeSubhead2(color: color ?? .themeGray) + } + } + } +} diff --git a/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupManager/BackupManagerModule.swift b/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupManager/BackupManagerModule.swift new file mode 100644 index 0000000000..785b0ea0be --- /dev/null +++ b/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupManager/BackupManagerModule.swift @@ -0,0 +1,7 @@ +import SwiftUI + +struct BackupManagerModule { + static func view() -> some View { + BackupManagerView() + } +} \ No newline at end of file diff --git a/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupManager/BackupManagerView.swift b/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupManager/BackupManagerView.swift new file mode 100644 index 0000000000..93c71f72c7 --- /dev/null +++ b/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupManager/BackupManagerView.swift @@ -0,0 +1,35 @@ +import SDWebImageSwiftUI +import SwiftUI +import ThemeKit + +struct BackupManagerView: View { + @State var restorePresented: Bool = false + @State var backupPresented: Bool = false + + var body: some View { + ScrollableThemeView { + ListSection { + ClickableRow(action: { + restorePresented = true + }) { + Image("download_24").themeIcon(color: .themeJacob) + Text("backup_manager.restore".localized).themeBody(color: .themeJacob) + } + ClickableRow(action: { + backupPresented = true + }) { + Image("plus_24").themeIcon(color: .themeJacob) + Text("backup_manager.create".localized).themeBody(color: .themeJacob) + } + } + .sheet(isPresented: $restorePresented) { + InfoModule.restoreSourceInfo + } + .sheet(isPresented: $backupPresented) { + ThemeNavigationView { BackupAppModule.view(backupPresented: $backupPresented) } + } + .navigationBarTitle("backup_manager.title".localized) + .padding(EdgeInsets(top: .margin12, leading: .margin16, bottom: .margin32, trailing: .margin16)) + } + } +} diff --git a/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupName/BackupNameView.swift b/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupName/BackupNameView.swift new file mode 100644 index 0000000000..5a32ef42e3 --- /dev/null +++ b/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupName/BackupNameView.swift @@ -0,0 +1,52 @@ +import SwiftUI +import ThemeKit + +struct BackupNameView: View { + @ObservedObject var viewModel: BackupAppViewModel + @Binding var backupPresented: Bool + + var body: some View { + ThemeView { + BottomGradientWrapper { + VStack(spacing: .margin24) { + Text("backup.name.description".localized) + .themeSubhead2() + .padding(EdgeInsets(top: 0, leading: .margin16, bottom: .margin12, trailing: .margin16)) + + InputTextRow { + InputTextView( + placeholder: "backup.cloud.name.placeholder".localized, + text: $viewModel.name + ) + .autocapitalization(.words) + .autocorrectionDisabled() + } + .modifier(CautionBorder(cautionState: $viewModel.nameCautionState)) + .modifier(CautionPrompt(cautionState: $viewModel.nameCautionState)) + } + .animation(.default, value: viewModel.nameCautionState) + .padding(EdgeInsets(top: .margin12, leading: .margin16, bottom: .margin32, trailing: .margin16)) + } bottomContent: { + NavigationLink( + destination: BackupPasswordView(viewModel: viewModel, backupPresented: $backupPresented), + isActive: $viewModel.passwordPushed + ) { + Button(action: { + viewModel.passwordPushed = true + }) { + Text("button.next".localized) + } + .buttonStyle(PrimaryButtonStyle(style: .yellow)) + .disabled(viewModel.nameCautionState != .none) + } + } + .navigationBarTitle("backup.name.title".localized) + .navigationBarTitleDisplayMode(.inline) + .toolbar { + Button("button.cancel".localized) { + backupPresented = false + } + } + } + } +} diff --git a/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupPassword/BackupPasswordView.swift b/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupPassword/BackupPasswordView.swift new file mode 100644 index 0000000000..e8d71c289e --- /dev/null +++ b/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupPassword/BackupPasswordView.swift @@ -0,0 +1,109 @@ +import SwiftUI +import ThemeKit +import ComponentKit + +struct BackupPasswordView: View { + @ObservedObject var viewModel: BackupAppViewModel + @Binding var backupPresented: Bool + + @State var secureLock = true + + var body: some View { + ThemeView { + BottomGradientWrapper { + VStack(spacing: .margin32) { + Text("backup.password.description".localized) + .themeSubhead2() + .padding(EdgeInsets(top: 0, leading: .margin16, bottom: 0, trailing: .margin16)) + + VStack(spacing: .margin16) { + InputTextRow { + InputTextView( + placeholder: "backup.cloud.password.placeholder".localized, + text: $viewModel.password, + isValidText: { text in PassphraseValidator.validate(text: text) } + ) + .secure($secureLock) + .autocapitalization(.none) + .autocorrectionDisabled() + } + .modifier(CautionBorder(cautionState: $viewModel.passwordCautionState)) + .modifier(CautionPrompt(cautionState: $viewModel.passwordCautionState)) + + InputTextRow { + InputTextView( + placeholder: "backup.cloud.password.confirm.placeholder".localized, + text: $viewModel.confirm, + isValidText: { text in PassphraseValidator.validate(text: text) } + ) + .secure($secureLock) + .autocapitalization(.none) + .autocorrectionDisabled() + } + .modifier(CautionBorder(cautionState: $viewModel.confirmCautionState)) + .modifier(CautionPrompt(cautionState: $viewModel.confirmCautionState)) + } + .animation(.default, value: secureLock) + + HighlightedTextView(text: "backup.password.highlighted_description".localized, style: .warning) + } + .animation(.default, value: viewModel.passwordCautionState) + .animation(.default, value: viewModel.confirmCautionState) + .padding(EdgeInsets(top: .margin12, leading: .margin16, bottom: .margin32, trailing: .margin16)) + } bottomContent: { + Button(action: { + viewModel.onTapSave() + }) { + HStack(spacing: .margin8) { + if viewModel.passwordButtonProcessing { + ProgressView().progressViewStyle(.circular) + } + + Text("button.save".localized) + } + } + .buttonStyle(PrimaryButtonStyle(style: .yellow)) + .disabled(viewModel.passwordButtonDisabled || viewModel.passwordButtonProcessing) + .animation(.default, value: viewModel.passwordButtonProcessing) + } + .sheet(item: $viewModel.sharePresented) { url in + let completion: UIActivityViewController.CompletionWithItemsHandler = { type, success, list, error in + if success { + showDone() + backupPresented = false + } + if let error { + show(error: error) + } + } + if #available(iOS 16, *) { + ActivityViewController(activityItems: [url], completionWithItemsHandler: completion).presentationDetents([.medium, .large]) + } else { + ActivityViewController(activityItems: [url], completionWithItemsHandler: completion) + } + } + .navigationBarTitle("backup.password.title".localized) + .navigationBarTitleDisplayMode(.inline) + .toolbar { + Button("button.cancel".localized) { + backupPresented = false + } + } + } + } + + @MainActor + private func showSuccess() { + HudHelper.instance.show(banner: .savedToCloud) + } + + @MainActor + private func show(error: Error) { + HudHelper.instance.show(banner: .error(string: error.localizedDescription)) + } + + @MainActor + func showDone() { + HudHelper.instance.show(banner: .done) + } +} diff --git a/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupType/BackupTypeView.swift b/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupType/BackupTypeView.swift new file mode 100644 index 0000000000..aee60de716 --- /dev/null +++ b/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BackupApp/BackupType/BackupTypeView.swift @@ -0,0 +1,69 @@ +import SwiftUI +import ThemeKit + +struct BackupTypeView: View { + @ObservedObject var viewModel: BackupAppViewModel + @Binding var backupPresented: Bool + + @State var navigationPushed = false + @State var cloudAlertPresented = false + + var body: some View { + ScrollableThemeView { + VStack(spacing: .margin24) { + Text("backup_type.description".localized) + .themeSubhead2() + .padding(EdgeInsets(top: 0, leading: .margin16, bottom: .margin12, trailing: .margin16)) + + ListSection { + navigation(image: "icloud_24", text: "backup_type.cloud".localized, isAvailable: $viewModel.cloudAvailable) { + if viewModel.cloudAvailable { viewModel.destination = .cloud } else { cloudAlertPresented = true } + } + + navigation(image: "file_24", text: "backup_type.file".localized) { + viewModel.destination = .local + } + } + .sheet(isPresented: $cloudAlertPresented) { + if #available(iOS 16, *) { + ViewWrapper(BottomSheetModule.cloudNotAvailableController()).presentationDetents([.medium]) + } else { + ViewWrapper(BottomSheetModule.cloudNotAvailableController()) + } + } + } + .navigationBarTitle("backup_type.title".localized) + .navigationBarTitleDisplayMode(.inline) + .toolbar { + Button("button.cancel".localized) { + backupPresented = false + } + } + .padding(EdgeInsets(top: .margin12, leading: .margin16, bottom: .margin32, trailing: .margin16)) + } + } + + @ViewBuilder func row(image: String, text: String) -> some View { + HStack(spacing: .margin16) { + Image(image).themeIcon() + Text(text).themeBody() + Image.disclosureIcon + } + } + + @ViewBuilder func navigation(image: String, text: String, isAvailable: Binding = .constant(true), action: @escaping () -> Void = {}) -> some View { + if isAvailable.wrappedValue { + NavigationRow( + destination: { BackupListView(viewModel: viewModel, backupPresented: $backupPresented) }, + isActive: $navigationPushed + ) { + row(image: image, text: text.localized) + } + .onChange(of: navigationPushed) { _ in action() } + } else { + ClickableRow(action: action) { + row(image: image, text: text.localized) + } + } + } +} diff --git a/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BlockchainSettings/BlockchainSettingsViewModel.swift b/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BlockchainSettings/BlockchainSettingsViewModel.swift index 317b1b6eed..55fc10cb99 100644 --- a/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BlockchainSettings/BlockchainSettingsViewModel.swift +++ b/UnstoppableWallet/UnstoppableWallet/Modules/Settings/BlockchainSettings/BlockchainSettingsViewModel.swift @@ -9,8 +9,8 @@ class BlockchainSettingsViewModel: ObservableObject { private let evmSyncSourceManager: EvmSyncSourceManager private let disposeBag = DisposeBag() - @Published var btcItems: [BtcItem] = [] @Published var evmItems: [EvmItem] = [] + @Published var btcItems: [BtcItem] = [] init(btcBlockchainManager: BtcBlockchainManager, evmBlockchainManager: EvmBlockchainManager, evmSyncSourceManager: EvmSyncSourceManager) { self.btcBlockchainManager = btcBlockchainManager diff --git a/UnstoppableWallet/UnstoppableWallet/Modules/Settings/Main/MainSettingsViewController.swift b/UnstoppableWallet/UnstoppableWallet/Modules/Settings/Main/MainSettingsViewController.swift index 30494b97f4..ca2b3a3f78 100644 --- a/UnstoppableWallet/UnstoppableWallet/Modules/Settings/Main/MainSettingsViewController.swift +++ b/UnstoppableWallet/UnstoppableWallet/Modules/Settings/Main/MainSettingsViewController.swift @@ -219,11 +219,22 @@ class MainSettingsViewController: ThemeViewController { image: .local(UIImage(named: "blocks_24")), title: .body("settings.blockchain_settings".localized), accessoryType: .disclosure, - isLast: true, + isLast: false, action: { [weak self] in let viewController = BlockchainSettingsModule.view().toViewController(title: "blockchain_settings.title".localized) self?.navigationController?.pushViewController(viewController, animated: true) } + ), + tableView.universalRow48( + id: "backup-manager", + image: .local(UIImage(named: "icloud_24")), + title: .body("settings.backup_manager".localized), + accessoryType: .disclosure, + isLast: true, + action: { [weak self] in + let viewController = BackupManagerModule.view().toViewController(title: "backup_manager.title".localized) + self?.navigationController?.pushViewController(viewController, animated: true) + } ) ] } diff --git a/UnstoppableWallet/UnstoppableWallet/UserInterface/Caution.swift b/UnstoppableWallet/UnstoppableWallet/UserInterface/Caution.swift index 8fd7187194..ce608b8d2e 100644 --- a/UnstoppableWallet/UnstoppableWallet/UserInterface/Caution.swift +++ b/UnstoppableWallet/UnstoppableWallet/UserInterface/Caution.swift @@ -1,7 +1,30 @@ -import UIKit +import SwiftUI import ThemeKit -struct Caution { +enum CautionState: Equatable { + case none + case caution(Caution) + + var caution: Caution? { + switch self { + case let .caution(caution): return caution + default: return nil + } + } + + var color: Color { + switch self { + case .none: return Color.clear + case let .caution(caution): + switch caution.type { + case .warning: return .themeJacob + case .error: return .themeLucian + } + } + } +} + +struct Caution: Equatable { let text: String let type: CautionType } @@ -24,13 +47,12 @@ enum CautionType: Equatable { } } - static func ==(lhs: CautionType, rhs: CautionType) -> Bool { + static func == (lhs: CautionType, rhs: CautionType) -> Bool { switch (lhs, rhs) { case (.error, .error), (.warning, .warning): return true default: return false } } - } class TitledCaution: Equatable { @@ -44,12 +66,11 @@ class TitledCaution: Equatable { self.type = type } - static func ==(lhs: TitledCaution, rhs: TitledCaution) -> Bool { + static func == (lhs: TitledCaution, rhs: TitledCaution) -> Bool { lhs.title == rhs.title && - lhs.text == rhs.text && - lhs.type == rhs.type + lhs.text == rhs.text && + lhs.type == rhs.type } - } class CancellableTitledCaution: TitledCaution { @@ -60,5 +81,4 @@ class CancellableTitledCaution: TitledCaution { super.init(title: title, text: text, type: type) } - } diff --git a/UnstoppableWallet/UnstoppableWallet/UserInterface/SwiftUI/CheckboxStyle.swift b/UnstoppableWallet/UnstoppableWallet/UserInterface/SwiftUI/CheckboxStyle.swift new file mode 100644 index 0000000000..4ff8103098 --- /dev/null +++ b/UnstoppableWallet/UnstoppableWallet/UserInterface/SwiftUI/CheckboxStyle.swift @@ -0,0 +1,23 @@ +import SwiftUI +import ThemeKit + +struct CheckboxStyle: ToggleStyle { + private let size: CGFloat = .margin24 - .heightOneDp + + func makeBody(configuration: Configuration) -> some View { + Button(action: { + configuration.isOn.toggle() + }, label: { + Image("check_2_20") + .themeIcon(color: .themeJacob) + .opacity(configuration.isOn ? 1 : 0) + .frame(width: size, height: size, alignment: .center) + }) + .overlay( + RoundedRectangle(cornerRadius: .cornerRadius4, style: .continuous) + .stroke(Color.themeGray, lineWidth: .heightOneDp + .heightOnePixel) + ) + + configuration.label + } +} diff --git a/UnstoppableWallet/UnstoppableWallet/UserInterface/SwiftUI/Extensions/ActivityViewController.swift b/UnstoppableWallet/UnstoppableWallet/UserInterface/SwiftUI/Extensions/ActivityViewController.swift new file mode 100644 index 0000000000..cba5720d66 --- /dev/null +++ b/UnstoppableWallet/UnstoppableWallet/UserInterface/SwiftUI/Extensions/ActivityViewController.swift @@ -0,0 +1,22 @@ +import SwiftUI +import UIKit + +struct ActivityViewController: UIViewControllerRepresentable { + var activityItems: [Any] + var applicationActivities: [UIActivity]? = nil + var completionWithItemsHandler: UIActivityViewController.CompletionWithItemsHandler? + + func makeUIViewController(context _: UIViewControllerRepresentableContext) -> UIActivityViewController { + let controller = UIActivityViewController(activityItems: activityItems, applicationActivities: applicationActivities) + controller.completionWithItemsHandler = completionWithItemsHandler + return controller + } + + func updateUIViewController(_: UIActivityViewController, context _: UIViewControllerRepresentableContext) {} +} + +extension URL: Identifiable { + public var id: String { + absoluteString + } +} diff --git a/UnstoppableWallet/UnstoppableWallet/UserInterface/SwiftUI/Extensions/Text.swift b/UnstoppableWallet/UnstoppableWallet/UserInterface/SwiftUI/Extensions/Text.swift index 2b9030e947..f2550d4128 100644 --- a/UnstoppableWallet/UnstoppableWallet/UserInterface/SwiftUI/Extensions/Text.swift +++ b/UnstoppableWallet/UnstoppableWallet/UserInterface/SwiftUI/Extensions/Text.swift @@ -1,26 +1,27 @@ import SwiftUI extension Text { - func themeBody(color: Color = .themeLeah, alignment: Alignment = .leading) -> some View { - self - .frame(maxWidth: .infinity, alignment: alignment) - .foregroundColor(color) - .font(.themeBody) + frame(maxWidth: .infinity, alignment: alignment) + .foregroundColor(color) + .font(.themeBody) } func themeSubhead1(color: Color = .themeGray, alignment: Alignment = .leading) -> some View { - self - .frame(maxWidth: .infinity, alignment: alignment) - .foregroundColor(color) - .font(.themeSubhead1) + frame(maxWidth: .infinity, alignment: alignment) + .foregroundColor(color) + .font(.themeSubhead1) } func themeSubhead2(color: Color = .themeGray, alignment: Alignment = .leading) -> some View { - self - .frame(maxWidth: .infinity, alignment: alignment) - .foregroundColor(color) - .font(.themeSubhead2) + frame(maxWidth: .infinity, alignment: alignment) + .foregroundColor(color) + .font(.themeSubhead2) } + func themeCaption(color: Color = .themeGray, alignment: Alignment = .leading) -> some View { + frame(maxWidth: .infinity, alignment: alignment) + .foregroundColor(color) + .font(.themeCaption) + } } diff --git a/UnstoppableWallet/UnstoppableWallet/UserInterface/SwiftUI/InputTextRow.swift b/UnstoppableWallet/UnstoppableWallet/UserInterface/SwiftUI/InputTextRow.swift new file mode 100644 index 0000000000..b383757384 --- /dev/null +++ b/UnstoppableWallet/UnstoppableWallet/UserInterface/SwiftUI/InputTextRow.swift @@ -0,0 +1,15 @@ +import SwiftUI + +struct InputTextRow: View { + @ViewBuilder let content: Content + + var body: some View { + HStack(spacing: .margin16) { + content + } + .padding(EdgeInsets(top: .margin12, leading: .margin16, bottom: .margin12, trailing: .margin16)) + .background(RoundedRectangle(cornerRadius: .cornerRadius8, style: .continuous).fill(Color.themeLawrence)) + .overlay(RoundedRectangle(cornerRadius: .cornerRadius8, style: .continuous).stroke(Color.themeSteel20, lineWidth: .heightOneDp)) + .frame(minHeight: .heightSingleLineCell) + } +} diff --git a/UnstoppableWallet/UnstoppableWallet/UserInterface/SwiftUI/InputTextView.swift b/UnstoppableWallet/UnstoppableWallet/UserInterface/SwiftUI/InputTextView.swift new file mode 100644 index 0000000000..77e076ea1d --- /dev/null +++ b/UnstoppableWallet/UnstoppableWallet/UserInterface/SwiftUI/InputTextView.swift @@ -0,0 +1,125 @@ +import SwiftUI +import ThemeKit + +struct InputTextView: View { + var placeholder: String = "" + + var text: Binding + @State private var oldValue: String + + @Binding var secured: Bool + + @State var shake = false + var shakeOnInvalid = true + + var onEditingChanged: ((Bool) -> Void)? + var onCommit: (() -> Void)? + var isValidText: ((String) -> Bool)? + + init(placeholder: String = "", text: Binding, secured: Binding = .constant(false), onEditingChanged: ((Bool) -> Void)? = nil, onCommit: (() -> Void)? = nil, isValidText: ((String) -> Bool)? = nil) { + self.placeholder = placeholder + self.text = text + oldValue = text.wrappedValue + self._secured = secured + + self.onEditingChanged = onEditingChanged + self.onCommit = onCommit + self.isValidText = isValidText + } + + var body: some View { + editView() + .font(.themeBody) + .accentColor(.themeLeah) + .frame(height: 20) //todo: How to remove this? + .onReceive(text.wrappedValue.publisher.collect()) { + let newValue = $0.map { String($0) }.joined() + if isValidText?(newValue) ?? true { + oldValue = newValue + } else { + text.wrappedValue = oldValue + + if shakeOnInvalid { + shake = true + } + } + } + .shake($shake) + } + + @ViewBuilder + func editView() -> some View { + if secured { + SecureField( + placeholder, + text: text, + onCommit: { commit() } + ) + } else { + TextField( + placeholder, + text: text, + onEditingChanged: { editingChanged($0) }, + onCommit: { commit() } + ) + } + } + + private func editingChanged(_ bool: Bool) { + onEditingChanged?(bool) + } + + private func commit() { + onCommit?() + } +} + +extension InputTextView { + @ViewBuilder func secure(_ secured: Binding) -> some View { + var selfView = self + selfView._secured = secured + + return HStack(spacing: .margin16) { + selfView + + Button(action: { + secured.wrappedValue.toggle() + }) { + Image(secured.wrappedValue ? "eye_off_20" : "eye_20").themeIcon() + } + } + } +} + +struct CautionBorder: ViewModifier { + let cornerRadius: CGFloat + @Binding var cautionState: CautionState + + init(cornerRadius: CGFloat = .cornerRadius8, cautionState: Binding) { + self.cornerRadius = cornerRadius + _cautionState = cautionState + } + + func body(content: Content) -> some View { + content + .overlay( + RoundedRectangle(cornerRadius: cornerRadius, style: .continuous) + .stroke(cautionState.color, lineWidth: .heightOneDp) + ) + } +} + +struct CautionPrompt: ViewModifier { + @Binding var cautionState: CautionState + + func body(content: Content) -> some View { + VStack { + content + + if let caution = cautionState.caution { + Text(caution.text) + .themeCaption(color: cautionState.color) + } + } + } +} diff --git a/UnstoppableWallet/UnstoppableWallet/UserInterface/SwiftUI/NavigationRow.swift b/UnstoppableWallet/UnstoppableWallet/UserInterface/SwiftUI/NavigationRow.swift index deb5523386..7c39a09b58 100644 --- a/UnstoppableWallet/UnstoppableWallet/UserInterface/SwiftUI/NavigationRow.swift +++ b/UnstoppableWallet/UnstoppableWallet/UserInterface/SwiftUI/NavigationRow.swift @@ -2,14 +2,19 @@ import SwiftUI struct NavigationRow: View { @ViewBuilder let destination: Destination + var isActive: Binding? @ViewBuilder let content: Content var body: some View { - NavigationLink(destination: destination) { - ListRow { - content - } + let row = ListRow { + content + } + if let isActive { + NavigationLink(destination: destination, isActive: isActive) { row } + .buttonStyle(RowButtonStyle()) + } else { + NavigationLink(destination: destination) { row } + .buttonStyle(RowButtonStyle()) } - .buttonStyle(RowButtonStyle()) } } diff --git a/UnstoppableWallet/UnstoppableWallet/UserInterface/SwiftUI/Shake.swift b/UnstoppableWallet/UnstoppableWallet/UserInterface/SwiftUI/Shake.swift new file mode 100644 index 0000000000..7c6c3a63f4 --- /dev/null +++ b/UnstoppableWallet/UnstoppableWallet/UserInterface/SwiftUI/Shake.swift @@ -0,0 +1,86 @@ +import SwiftUI + +struct Shake: View { + /// Set to true in order to animate + @Binding var shake: Bool + /// How many times the content will animate back and forth + var repeatCount = 3 + /// Duration in seconds + var duration = 0.4 + /// Range in pixels to go back and forth + var offsetRange = -5.0 + + @ViewBuilder let content: Content + var onCompletion: (() -> Void)? + + @State private var xOffset = 0.0 + + var body: some View { + content + .offset(x: xOffset) + .onChange(of: shake) { shouldShake in + guard shouldShake else { return } + Task { + await animate() + shake = false + onCompletion?() + } + } + } + + // Obs: some of factors must be 1.0. + private func animate() async { + let factor1 = 0.9 + let eachDuration = duration * factor1 / CGFloat(repeatCount) + for _ in 0 ..< repeatCount { + await backAndForthAnimation(duration: eachDuration, offset: offsetRange) + } + + let factor2 = 0.1 + await animate(duration: duration * factor2) { + xOffset = 0.0 + } + } + + private func backAndForthAnimation(duration: CGFloat, offset: CGFloat) async { + let halfDuration = duration / 2 + await animate(duration: halfDuration) { + self.xOffset = offset + } + + await animate(duration: halfDuration) { + self.xOffset = -offset + } + } +} + +extension View { + func shake(_ shake: Binding, + repeatCount: Int = 3, + duration: CGFloat = 0.4, + offsetRange: CGFloat = -5, + onCompletion: (() -> Void)? = nil) -> some View + { + Shake(shake: shake, + repeatCount: repeatCount, + duration: duration, + offsetRange: offsetRange) + { + self + } onCompletion: { + onCompletion?() + } + } + + func animate(duration: CGFloat, _ execute: @escaping () -> Void) async { + await withCheckedContinuation { continuation in + withAnimation(.linear(duration: duration)) { + execute() + } + + DispatchQueue.main.asyncAfter(deadline: .now() + duration) { + continuation.resume() + } + } + } +} diff --git a/UnstoppableWallet/UnstoppableWallet/en.lproj/Localizable.strings b/UnstoppableWallet/UnstoppableWallet/en.lproj/Localizable.strings index 84030fcf24..b5a4cee487 100644 --- a/UnstoppableWallet/UnstoppableWallet/en.lproj/Localizable.strings +++ b/UnstoppableWallet/UnstoppableWallet/en.lproj/Localizable.strings @@ -12,6 +12,7 @@ "button.paste" = "Paste"; "button.resend" = "Resend"; "button.backup" = "Backup"; +"button.restore" = "Restore"; "button.copy" = "Copy"; "button.retry" = "Retry"; "button.report" = "Report"; @@ -1011,6 +1012,7 @@ Go to Settings - > %@ and allow access to the camera."; "settings.tab_bar_item" = "Settings"; "settings.manage_accounts" = "Manage Wallets"; "settings.blockchain_settings" = "Blockchain Settings"; +"settings.backup_manager" = "Backup Manager"; "settings.security" = "Security"; "settings.experimental_features" = "Experimental"; "settings.personal_support" = "Personal Support"; @@ -1083,6 +1085,46 @@ Go to Settings - > %@ and allow access to the camera."; "blockchain_settings.title" = "Blockchain Settings"; +// Settings -> Backup Manager + +"backup_manager.title" = "Backup Manager"; +"backup_manager.restore" = "Restore Backup"; +"backup_manager.create" = "Create New Backup"; + +"backup_type.title" = "Backup Type"; +"backup_type.description" = "Select where you want to save the backup file."; +"backup_type.cloud" = "Backup to iCloud"; +"backup_type.file" = "Backup to Files"; + +"backup_list.title" = "Backup File"; +"backup_list.description.restore" = "List of contents in the backup file."; +"backup_list.header.wallets" = "Wallets"; +"backup_list.header.other" = "Other"; +"backup_list.other.watch_account.title" = "Watch Wallets"; +"backup_list.other.watch_account.description" = "Addresses: %d"; +"backup_list.other.watchlist.title" = "Watchlist"; +"backup_list.other.watchlist.description" = "Coins: %d"; +"backup_list.other.contacts.title" = "Contacts"; +"backup_list.other.contacts.description" = "Addresses: %d"; +"backup_list.other.blockchain_settings.title" = "Blockchain Settings"; +"backup_list.other.blockchain_settings.description" = "Custom RPCs: %d"; +"backup_list.other.app_settings.title" = "App Settings"; +"backup_list.other.app_settings.description" = "Language, Currency, Appearance ..."; + +"backup.disclaimer.cloud.title" = "Backup to iCloud"; +"backup.disclaimer.cloud.description" = "iCloud is a cloud storage service provided by Apple. It's important to know that your backup data will be stored on Apple's servers."; +"backup.disclaimer.cloud.checkbox_label" = "I understand that losing access to my iCloud, will result in loosing access to the backup of a respective wallet."; +"backup.disclaimer.file.title" = "Backup to File"; +"backup.disclaimer.file.description" = "Storage devices i.e. hard drives, USB drives , storage on smartphone etc. are all vulnerable to loss due to physical damage, theft or other unforeseen circumstances."; +"backup.disclaimer.file.checkbox_label" = "I understand that theft or damage of a backup device will result in loss of a backup to a respective wallet."; + +"backup.name.title" = "Backup Name"; +"backup.name.description" = "Enter name for the backup file."; + +"backup.password.title" = "Backup Password"; +"backup.password.description" = "Set unlock password for your backup. It must consist of at least 8 symbols and include at least one lowercase letter, uppercase letter, number and a special character."; +"backup.password.highlighted_description" = "This password is used to encrypt backup file of your wallet. It can't be recovered or reset if lost or forgotten."; + // Settings -> Security "settings_security.title" = "Security";