diff --git a/Kukai Mobile.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved b/Kukai Mobile.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved index a81c4c7e..2d7d5abd 100644 --- a/Kukai Mobile.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved +++ b/Kukai Mobile.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved @@ -70,7 +70,7 @@ "location" : "https://github.com/kukai-wallet/kukai-core-swift", "state" : { "branch" : "develop", - "revision" : "7342af3601139f45da065c67ca71aa060ecccb8f" + "revision" : "922e83b63c68ac275182ae2df5dbf7ce0c4a3ae9" } }, { diff --git a/Kukai Mobile/Controls/EnterAddressComponent/EnterAddressComponent.xib b/Kukai Mobile/Controls/EnterAddressComponent/EnterAddressComponent.xib index 8aaea3e9..248b0a72 100644 --- a/Kukai Mobile/Controls/EnterAddressComponent/EnterAddressComponent.xib +++ b/Kukai Mobile/Controls/EnterAddressComponent/EnterAddressComponent.xib @@ -1,8 +1,9 @@ - + - + + @@ -44,13 +45,13 @@ - + @@ -62,8 +63,8 @@ - - + @@ -230,10 +231,10 @@ - + - + @@ -265,7 +266,7 @@ - + diff --git a/Kukai Mobile/Controls/ScanViewController.swift b/Kukai Mobile/Controls/ScanViewController.swift index 3b4854cb..9d12661f 100644 --- a/Kukai Mobile/Controls/ScanViewController.swift +++ b/Kukai Mobile/Controls/ScanViewController.swift @@ -18,8 +18,8 @@ protocol ScanViewControllerDelegate: AnyObject { } class ScanViewController: UIViewController, AVCaptureMetadataOutputObjectsDelegate { - var captureSession: AVCaptureSession! - var previewLayer: AVCaptureVideoPreviewLayer! = AVCaptureVideoPreviewLayer() + var captureSession: AVCaptureSession? + var previewLayer: AVCaptureVideoPreviewLayer = AVCaptureVideoPreviewLayer() let scrollView = AutoScrollView() let titleLabel = UILabel() @@ -51,7 +51,7 @@ class ScanViewController: UIViewController, AVCaptureMetadataOutputObjectsDelega AVCaptureDevice.requestAccess(for: .video) { [weak self] (response) in DispatchQueue.main.async { if response { - self?.setupVideoPreview() + self?.setupCaptureSession() } else { self?.failed() } @@ -73,10 +73,11 @@ class ScanViewController: UIViewController, AVCaptureMetadataOutputObjectsDelega textfield.text = nil + setupPreviewLayer() if (captureSession?.isRunning == false) { // Xcode warning, should be run on a background thread in order to avoid hanging UI thread DispatchQueue.global(qos: .background).async { [weak self] in - self?.captureSession.startRunning() + self?.captureSession?.startRunning() } } @@ -91,7 +92,7 @@ class ScanViewController: UIViewController, AVCaptureMetadataOutputObjectsDelega if (captureSession?.isRunning == true) { DispatchQueue.global(qos: .background).async { [weak self] in - self?.captureSession.stopRunning() + self?.captureSession?.stopRunning() } } } @@ -271,7 +272,7 @@ class ScanViewController: UIViewController, AVCaptureMetadataOutputObjectsDelega setupClearBox() } - func setupVideoPreview() { + func setupCaptureSession() { captureSession = AVCaptureSession() guard let videoCaptureDevice = AVCaptureDevice.default(for: .video) else { return } @@ -283,8 +284,8 @@ class ScanViewController: UIViewController, AVCaptureMetadataOutputObjectsDelega return } - if (captureSession.canAddInput(videoInput)) { - captureSession.addInput(videoInput) + if captureSession?.canAddInput(videoInput) == true { + captureSession?.addInput(videoInput) } else { failed() return @@ -292,8 +293,8 @@ class ScanViewController: UIViewController, AVCaptureMetadataOutputObjectsDelega let metadataOutput = AVCaptureMetadataOutput() - if (captureSession.canAddOutput(metadataOutput)) { - captureSession.addOutput(metadataOutput) + if captureSession?.canAddOutput(metadataOutput) == true { + captureSession?.addOutput(metadataOutput) metadataOutput.setMetadataObjectsDelegate(self, queue: DispatchQueue.main) metadataOutput.metadataObjectTypes = [.qr] @@ -301,17 +302,18 @@ class ScanViewController: UIViewController, AVCaptureMetadataOutputObjectsDelega failed() return } + } + + func setupPreviewLayer() { + guard let session = captureSession else { + return + } - previewLayer = AVCaptureVideoPreviewLayer(session: captureSession) + previewLayer = AVCaptureVideoPreviewLayer(session: session) previewLayer.frame = previewContainerView.bounds previewLayer.videoGravity = .resizeAspectFill previewContainerView.layer.insertSublayer(previewLayer, at: 0) - // Xcode warning, should be run on a background thread in order to avoid hanging UI thread - DispatchQueue.global(qos: .background).async { [weak self] in - self?.captureSession.startRunning() - } - view.setNeedsLayout() } @@ -321,9 +323,9 @@ class ScanViewController: UIViewController, AVCaptureMetadataOutputObjectsDelega if permissionsDenied { os_log(.error, log: .default, "User revoked camera permissions") - alertController = UIAlertController(title: "error", message: "NSCameraUsageDescription", preferredStyle: .alert) + alertController = UIAlertController(title: "error".localized(), message: "This app does not have permission to access the camera. If you wish to scan a QRCode, please go to settings and enable camera access", preferredStyle: .alert) - let systemSettingsAction = UIAlertAction(title: "wlt_navigation_settings", style: .default) { (action) in + let systemSettingsAction = UIAlertAction(title: "Settings", style: .default) { (action) in guard let settingsUrl = URL(string: UIApplication.openSettingsURLString) else { return } @@ -333,13 +335,13 @@ class ScanViewController: UIViewController, AVCaptureMetadataOutputObjectsDelega } } - alertController?.addAction(UIAlertAction(title: "cancel", style: .default)) + alertController?.addAction(UIAlertAction(title: "Cancel", style: .default)) alertController?.addAction(systemSettingsAction) } else { os_log(.error, log: .default, "Unable to scan on this device") - alertController = UIAlertController(title: "error", message: "error_cant_scan", preferredStyle: .alert) - alertController?.addAction(UIAlertAction(title: "ok", style: .default)) + alertController = UIAlertController(title: "error".localized(), message: "Unbale to setup camera at this time. Please check camera access is enabled in settings", preferredStyle: .alert) + alertController?.addAction(UIAlertAction(title: "Ok", style: .default)) } if let ac = alertController { diff --git a/Kukai Mobile/Modules/Collectibles/CollectiblesDetailsViewModel.swift b/Kukai Mobile/Modules/Collectibles/CollectiblesDetailsViewModel.swift index 001be0ad..b704deb7 100644 --- a/Kukai Mobile/Modules/Collectibles/CollectiblesDetailsViewModel.swift +++ b/Kukai Mobile/Modules/Collectibles/CollectiblesDetailsViewModel.swift @@ -317,6 +317,9 @@ class CollectiblesDetailsViewModel: ViewModel, UICollectionViewDiffableDataSourc } if self.attributes.count > 0 { + self.attributes = self.attributes.sorted { lhs, rhs in + return lhs.name < rhs.name + } self.currentSnapshot.insertItems([self.attributesContent], afterItem: self.descriptionData) self.currentSnapshot.appendItems(self.attributes, toSection: 1) needsUpdating = true diff --git a/Kukai Mobile/Modules/Stake/StakeViewController.swift b/Kukai Mobile/Modules/Stake/StakeViewController.swift index 8d4ee34a..9c8db00f 100644 --- a/Kukai Mobile/Modules/Stake/StakeViewController.swift +++ b/Kukai Mobile/Modules/Stake/StakeViewController.swift @@ -51,6 +51,8 @@ class StakeViewController: UIViewController { } public func enteredCustomBaker(address: String) { + self.showLoadingView() + if address == "" { let currentDelegate = DependencyManager.shared.balanceService.account.delegate let name = currentDelegate?.alias ?? currentDelegate?.address.truncateTezosAddress() ?? "" @@ -66,15 +68,25 @@ class StakeViewController: UIViewController { TransactionService.shared.delegateData.isAdd = false } else { - let baker = TzKTBaker(address: address, name: address.truncateTezosAddress(), logo: nil) - TransactionService.shared.delegateData.chosenBaker = baker - TransactionService.shared.delegateData.isAdd = true + self.showLoadingView() + + if let foundBaker = viewModel.bakerFor(address: address) { + TransactionService.shared.delegateData.chosenBaker = foundBaker + TransactionService.shared.delegateData.isAdd = true + + } else { + let baker = TzKTBaker(address: address, name: address.truncateTezosAddress(), logo: nil) + TransactionService.shared.delegateData.chosenBaker = baker + TransactionService.shared.delegateData.isAdd = true + } } createOperationsAndConfirm(toAddress: address) } public func stakeTapped() { + self.showLoadingView() + if let baker = TransactionService.shared.delegateData.chosenBaker { createOperationsAndConfirm(toAddress: baker.address) } @@ -86,7 +98,6 @@ class StakeViewController: UIViewController { return } - self.showLoadingView() let operations = OperationFactory.delegateOperation(to: toAddress, from: selectedWallet.address) DependencyManager.shared.tezosNodeClient.estimate(operations: operations, walletAddress: selectedWallet.address, base58EncodedPublicKey: selectedWallet.publicKeyBase58encoded()) { [weak self] estimationResult in self?.hideLoadingView() diff --git a/Kukai Mobile/Modules/Stake/StakeViewModel.swift b/Kukai Mobile/Modules/Stake/StakeViewModel.swift index 328cb47b..40a2f6e8 100644 --- a/Kukai Mobile/Modules/Stake/StakeViewModel.swift +++ b/Kukai Mobile/Modules/Stake/StakeViewModel.swift @@ -24,6 +24,7 @@ class StakeViewModel: ViewModel, UITableViewDiffableDataSourceHandler { private var currentSnapshot = NSDiffableDataSourceSnapshot() var dataSource: UITableViewDiffableDataSource? = nil + var bakers: [TzKTBaker] = [] @@ -93,6 +94,7 @@ class StakeViewModel: ViewModel, UITableViewDiffableDataSourceHandler { return } + self?.bakers = res let filteredResults = res.filter { baker in if baker.address == currentDelegate?.address { currentBaker = baker @@ -152,6 +154,10 @@ class StakeViewModel: ViewModel, UITableViewDiffableDataSourceHandler { return dataSource?.itemIdentifier(for: indexPath) as? TzKTBaker } + func bakerFor(address: String) -> TzKTBaker? { + return bakers.first(where: { $0.address == address }) + } + func isEnterCustom(indePath: IndexPath) -> Bool { if let obj = dataSource?.itemIdentifier(for: indePath) as? StakeHeaderData { return obj.actionTitle != nil