diff --git a/.periphery.yml b/.periphery.yml index 58b02ba2450..00c4eb71840 100644 --- a/.periphery.yml +++ b/.periphery.yml @@ -50,41 +50,41 @@ targets: # - PaymentSheetLocalizationScreenshotGenerator # - PaymentSheetUITest - Stripe3DS2 - - Stripe3DS2Tests + # - Stripe3DS2Tests - StripeApplePay - - StripeApplePayTests + # - StripeApplePayTests - StripeCameraCore - - StripeCameraCoreTestUtils - - StripeCameraCoreTests + # - StripeCameraCoreTestUtils + # - StripeCameraCoreTests - StripeCardScan - - StripeCardScanTests + # - StripeCardScanTests - StripeConnect # - StripeConnect Example # - StripeConnect ExampleUITests - - StripeConnectTests + # - StripeConnectTests - StripeCore - - StripeCoreTestUtils - - StripeCoreTests + # - StripeCoreTestUtils + # - StripeCoreTests - StripeFinancialConnections - - StripeFinancialConnectionsTests + # - StripeFinancialConnectionsTests - StripeIdentity - - StripeIdentityTests + # - StripeIdentityTests - StripePaymentSheet - - StripePaymentSheetTestHostApp - - StripePaymentSheetTests + # - StripePaymentSheetTestHostApp + # - StripePaymentSheetTests - StripePayments - - StripePaymentsObjcTestUtils - - StripePaymentsTestHostApp - - StripePaymentsTestUtils - - StripePaymentsTests + # - StripePaymentsObjcTestUtils + # - StripePaymentsTestHostApp + # - StripePaymentsTestUtils + # - StripePaymentsTests - StripePaymentsUI - - StripePaymentsUITests + # - StripePaymentsUITests - StripeUICore - - StripeUICoreTests + # - StripeUICoreTests - StripeiOS - - StripeiOSAppHostedTests - - StripeiOSTestHostApp - - StripeiOSTests + # - StripeiOSAppHostedTests + # - StripeiOSTestHostApp + # - StripeiOSTests # - UI Examples retain_public: true diff --git a/Example/UI Examples/UI Examples/Source/BrowseViewController.swift b/Example/UI Examples/UI Examples/Source/BrowseViewController.swift index a9ada5fcbdf..801e052440f 100644 --- a/Example/UI Examples/UI Examples/Source/BrowseViewController.swift +++ b/Example/UI Examples/UI Examples/Source/BrowseViewController.swift @@ -13,18 +13,14 @@ import UIKit @testable import Stripe @_spi(STP) import StripePaymentsUI -class BrowseViewController: UITableViewController, STPAddCardViewControllerDelegate, - STPPaymentOptionsViewControllerDelegate, STPShippingAddressViewControllerDelegate +class BrowseViewController: UITableViewController { enum Demo: Int { - static var count: Int = 11 + static var count: Int = 8 case STPPaymentCardTextField case STPPaymentCardTextFieldWithCBC - case STPPaymentOptionsViewController - case STPPaymentOptionsFPXViewController - case STPShippingInfoViewController case STPAUBECSFormViewController case STPCardFormViewController case STPCardFormViewControllerCBC @@ -36,9 +32,6 @@ class BrowseViewController: UITableViewController, STPAddCardViewControllerDeleg switch self { case .STPPaymentCardTextField: return "Card Field" case .STPPaymentCardTextFieldWithCBC: return "Card Field (CBC)" - case .STPPaymentOptionsViewController: return "Payment Option Picker" - case .STPPaymentOptionsFPXViewController: return "Payment Option Picker (With FPX)" - case .STPShippingInfoViewController: return "Shipping Info Form" case .STPAUBECSFormViewController: return "AU BECS Form" case .STPCardFormViewController: return "Card Form" case .STPCardFormViewControllerCBC: return "Card Form (CBC)" @@ -52,9 +45,6 @@ class BrowseViewController: UITableViewController, STPAddCardViewControllerDeleg switch self { case .STPPaymentCardTextField: return "STPPaymentCardTextField" case .STPPaymentCardTextFieldWithCBC: return "STPPaymentCardTextField" - case .STPPaymentOptionsViewController: return "STPPaymentOptionsViewController" - case .STPPaymentOptionsFPXViewController: return "STPPaymentOptionsViewController" - case .STPShippingInfoViewController: return "STPShippingInfoViewController" case .STPAUBECSFormViewController: return "STPAUBECSFormViewController" case .STPCardFormViewController: return "STPCardFormViewController" case .STPCardFormViewControllerCBC: return "STPCardFormViewController (CBC)" @@ -65,13 +55,6 @@ class BrowseViewController: UITableViewController, STPAddCardViewControllerDeleg } } - let customerContext: MockCustomerContext = { - let keyManager = STPEphemeralKeyManager( - keyProvider: MockKeyProvider(), - apiVersion: STPAPIClient.apiVersion, - performsEagerFetching: true) - return MockCustomerContext(keyManager: keyManager, apiClient: .shared) - }() let themeViewController = ThemeViewController() override func viewDidLoad() { @@ -121,47 +104,6 @@ class BrowseViewController: UITableViewController, STPAddCardViewControllerDeleg let navigationController = UINavigationController(rootViewController: viewController) navigationController.navigationBar.stp_theme = theme present(navigationController, animated: true, completion: nil) - case .STPPaymentOptionsFPXViewController: - let config = STPPaymentConfiguration() - config.fpxEnabled = true - config.requiredBillingAddressFields = .none - config.appleMerchantIdentifier = "dummy-merchant-id" - config.cardScanningEnabled = true - let viewController = STPPaymentOptionsViewController( - configuration: config, - theme: theme, - customerContext: self.customerContext, - delegate: self) - let navigationController = UINavigationController(rootViewController: viewController) - navigationController.navigationBar.stp_theme = theme - present(navigationController, animated: true, completion: nil) - case .STPPaymentOptionsViewController: - let config = STPPaymentConfiguration() - config.requiredBillingAddressFields = .none - config.appleMerchantIdentifier = "dummy-merchant-id" - config.cardScanningEnabled = true - let viewController = STPPaymentOptionsViewController( - configuration: config, - theme: theme, - customerContext: self.customerContext, - delegate: self) - let navigationController = UINavigationController(rootViewController: viewController) - navigationController.navigationBar.stp_theme = theme - present(navigationController, animated: true, completion: nil) - case .STPShippingInfoViewController: - let config = STPPaymentConfiguration() - config.requiredShippingAddressFields = [.postalAddress] - let viewController = STPShippingAddressViewController( - configuration: config, - theme: theme, - currency: "usd", - shippingAddress: nil, - selectedShippingMethod: nil, - prefilledInformation: nil) - viewController.delegate = self - let navigationController = UINavigationController(rootViewController: viewController) - navigationController.navigationBar.stp_theme = theme - present(navigationController, animated: true, completion: nil) case .STPAUBECSFormViewController: let viewController = AUBECSDebitFormViewController() viewController.theme = theme @@ -193,95 +135,4 @@ class BrowseViewController: UITableViewController, STPAddCardViewControllerDeleg } } - // MARK: STPAddCardViewControllerDelegate - - func addCardViewControllerDidCancel(_ addCardViewController: STPAddCardViewController) { - dismiss(animated: true, completion: nil) - } - - func addCardViewController( - _ addCardViewController: STPAddCardViewController, - didCreatePaymentMethod paymentMethod: STPPaymentMethod, - completion: @escaping STPErrorBlock - ) { - dismiss(animated: true, completion: nil) - } - - // MARK: STPPaymentOptionsViewControllerDelegate - - func paymentOptionsViewControllerDidCancel( - _ paymentOptionsViewController: STPPaymentOptionsViewController - ) { - dismiss(animated: true, completion: nil) - } - - func paymentOptionsViewControllerDidFinish( - _ paymentOptionsViewController: STPPaymentOptionsViewController - ) { - dismiss(animated: true, completion: nil) - } - - func paymentOptionsViewController( - _ paymentOptionsViewController: STPPaymentOptionsViewController, - didFailToLoadWithError error: Error - ) { - dismiss(animated: true, completion: nil) - } - - // MARK: STPShippingAddressViewControllerDelegate - - func shippingAddressViewControllerDidCancel( - _ addressViewController: STPShippingAddressViewController - ) { - dismiss(animated: true, completion: nil) - } - - func shippingAddressViewController( - _ addressViewController: STPShippingAddressViewController, - didFinishWith address: STPAddress, - shippingMethod method: PKShippingMethod? - ) { - self.customerContext.updateCustomer(withShippingAddress: address, completion: nil) - dismiss(animated: true, completion: nil) - } - - func shippingAddressViewController( - _ addressViewController: STPShippingAddressViewController, - didEnter address: STPAddress, - completion: @escaping STPShippingMethodsCompletionBlock - ) { - let upsGround = PKShippingMethod() - upsGround.amount = 0 - upsGround.label = "UPS Ground" - upsGround.detail = "Arrives in 3-5 days" - upsGround.identifier = "ups_ground" - let upsWorldwide = PKShippingMethod() - upsWorldwide.amount = 10.99 - upsWorldwide.label = "UPS Worldwide Express" - upsWorldwide.detail = "Arrives in 1-3 days" - upsWorldwide.identifier = "ups_worldwide" - let fedEx = PKShippingMethod() - fedEx.amount = 5.99 - fedEx.label = "FedEx" - fedEx.detail = "Arrives tomorrow" - fedEx.identifier = "fedex" - - DispatchQueue.main.asyncAfter(deadline: .now() + 0.6) { - if address.country == nil || address.country == "US" { - completion(.valid, nil, [upsGround, fedEx], fedEx) - } else if address.country == "AQ" { - let error = NSError( - domain: "ShippingError", code: 123, - userInfo: [ - NSLocalizedDescriptionKey: "Invalid Shipping Address", - NSLocalizedFailureReasonErrorKey: "We can't ship to this country.", - ]) - completion(.invalid, error, nil, nil) - } else { - fedEx.amount = 20.99 - completion(.valid, nil, [upsWorldwide, fedEx], fedEx) - } - } - } - } diff --git a/Example/UI Examples/UI Examples/Source/MockCustomerContext.swift b/Example/UI Examples/UI Examples/Source/MockCustomerContext.swift index 68095395ff3..311995e6a54 100644 --- a/Example/UI Examples/UI Examples/Source/MockCustomerContext.swift +++ b/Example/UI Examples/UI Examples/Source/MockCustomerContext.swift @@ -110,74 +110,3 @@ class MockKeyProvider: NSObject, STPCustomerEphemeralKeyProvider { completion(nil, NSError.stp_ephemeralKeyDecodingError()) } } - -class MockCustomerContext: STPCustomerContext { - - let customer = MockCustomer() - - override func retrieveCustomer(_ completion: STPCustomerCompletionBlock? = nil) { - if let completion = completion { - completion(customer, nil) - } - } - - override func attachPaymentMethod( - toCustomer paymentMethod: STPPaymentMethod, completion: STPErrorBlock? = nil - ) { - customer.paymentMethods.append(paymentMethod) - if let completion = completion { - completion(nil) - } - } - - override func detachPaymentMethod( - fromCustomer paymentMethod: STPPaymentMethod, completion: STPErrorBlock? = nil - ) { - if let index = customer.paymentMethods.firstIndex(where: { - $0.stripeId == paymentMethod.stripeId - }) { - customer.paymentMethods.remove(at: index) - } - if let completion = completion { - completion(nil) - } - } - - override func listPaymentMethodsForCustomer(completion: STPPaymentMethodsCompletionBlock? = nil) - { - if let completion = completion { - completion(customer.paymentMethods, nil) - } - } - - func selectDefaultCustomerPaymentMethod( - _ paymentMethod: STPPaymentMethod, completion: @escaping STPErrorBlock - ) { - if customer.paymentMethods.contains(where: { $0.stripeId == paymentMethod.stripeId }) { - customer.defaultPaymentMethod = paymentMethod - } - completion(nil) - } - - override func updateCustomer( - withShippingAddress shipping: STPAddress, completion: STPErrorBlock? - ) { - customer.shippingAddress = shipping - if let completion = completion { - completion(nil) - } - } - - override func retrieveLastSelectedPaymentMethodIDForCustomer( - completion: @escaping (String?, Error?) -> Void - ) { - completion(nil, nil) - } - override func saveLastSelectedPaymentMethodID( - forCustomer paymentMethodID: String?, completion: STPErrorBlock? - ) { - if let completion = completion { - completion(nil) - } - } -} diff --git a/Stripe/Stripe.xcodeproj/project.pbxproj b/Stripe/Stripe.xcodeproj/project.pbxproj index 1e6707b9a69..de4fd52880c 100644 --- a/Stripe/Stripe.xcodeproj/project.pbxproj +++ b/Stripe/Stripe.xcodeproj/project.pbxproj @@ -16,14 +16,12 @@ 07A5CDBFDF2340BAD99D6EB3 /* STPCardFormViewSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1D983E089196152DA1C69469 /* STPCardFormViewSnapshotTests.swift */; }; 07BF3CF1656AF5F5A0678873 /* STPPhoneNumberValidatorTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 924E878428D15506711CA628 /* STPPhoneNumberValidatorTest.swift */; }; 08111F4AD3CA0755420E05F7 /* LaunchScreen.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 147D2DC1FFDFC99269039377 /* LaunchScreen.storyboard */; }; - 08ED7A4EB7E64FDAED2C2D39 /* STPShippingAddressViewControllerTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 789D0B49B0788794739E3DD4 /* STPShippingAddressViewControllerTest.swift */; }; 093FE3D65978E3DB6B79AE05 /* UIToolbar+Stripe_InputAccessory.swift in Sources */ = {isa = PBXBuildFile; fileRef = E2307F2C5E53540D4ACAA1F6 /* UIToolbar+Stripe_InputAccessory.swift */; }; 0B9C0E9A7A750607413C9E53 /* STPFakeAddPaymentPassViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0669B4CA326CE74D125C789C /* STPFakeAddPaymentPassViewController.swift */; }; 0C5F4AE769D95AA921F61084 /* STPApplePayPaymentOptionTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 00985EFC6CB7B912FDBF3813 /* STPApplePayPaymentOptionTest.swift */; }; 0DFA17378D894C70D72C9F62 /* Error+PaymentSheetTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 8CE85C770AEEDBE4AEC93EAA /* Error+PaymentSheetTests.swift */; }; 0FA3C1494BA57884B5DE3B20 /* Stripe.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 4259421D2CD26E37B96F97B2 /* Stripe.framework */; }; 10342D659764A88A695EF38B /* Images.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = DFA7A75BA785EBBE4C05DAA3 /* Images.xcassets */; }; - 124D43C1A633922B1DA3E1E7 /* STPShippingAddressViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 71711FC8E2FB66E52A5FDD9A /* STPShippingAddressViewController.swift */; }; 14656D177E67594B8C75A9FE /* STPConnectAccountParamsTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 195DEC752CC82CC4BA1E2351 /* STPConnectAccountParamsTest.swift */; }; 162C101E57D66F0051164C4A /* Stripe.framework in Embed Frameworks */ = {isa = PBXBuildFile; fileRef = 4259421D2CD26E37B96F97B2 /* Stripe.framework */; settings = {ATTRIBUTES = (CodeSignOnCopy, RemoveHeadersOnCopy, ); }; }; 172D96526023A80534D54CC0 /* STPBankSelectionViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 81352A0CBE46A59E6B1A712E /* STPBankSelectionViewController.swift */; }; @@ -35,7 +33,6 @@ 1CCFC43F7FCD273E2100D321 /* STPPaymentMethodBancontactTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = F4E5416F6AE8BED88980D6F8 /* STPPaymentMethodBancontactTests.swift */; }; 1CD3AB315580606AF87A7B1F /* STPPaymentCardTextFieldKVOTest.m in Sources */ = {isa = PBXBuildFile; fileRef = BB08D2AC882B21C8ADD76B92 /* STPPaymentCardTextFieldKVOTest.m */; }; 1E8D8E2494062262A332879C /* STPCardValidatorTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = E1644AA33E81233EF33022BA /* STPCardValidatorTest.swift */; }; - 1F432D0B37949217E4299A20 /* STPPaymentOptionsInternalViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = D5C4A4CC7D2E9B5AB3EC3B79 /* STPPaymentOptionsInternalViewController.swift */; }; 225140E0BD9C0630116DDE4A /* STPPaymentMethodUSBankAccountTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3B112FFF3FCA82094281493F /* STPPaymentMethodUSBankAccountTest.swift */; }; 22BE2ABB29F77362FF16D945 /* Localizable.strings in Resources */ = {isa = PBXBuildFile; fileRef = C2427C1CDFA85BFC6570F1E9 /* Localizable.strings */; }; 234C71F480318E9062075924 /* AppDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = 9BBCE3A905041A709E8F279A /* AppDelegate.swift */; }; @@ -43,7 +40,6 @@ 23D1246A5DAB5333650F104F /* STPSectionHeaderView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 96E1FED5CE5974C9C1162E93 /* STPSectionHeaderView.swift */; }; 240993144289CD0DEC2C73C7 /* STPConfirmPaymentMethodOptionsTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 05FE1BA89B80336F16924FA2 /* STPConfirmPaymentMethodOptionsTest.swift */; }; 246920234EE8382FB4E56516 /* STPCardFormViewTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = FF5E08A1651D9DFE502DA021 /* STPCardFormViewTests.swift */; }; - 26F38A2A57FDDC12926BE044 /* STPAddCardViewControllerLocalizationSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 26E5C1DABAA63F07F0C6AE37 /* STPAddCardViewControllerLocalizationSnapshotTests.swift */; }; 279D2BA91198E18730626CE6 /* STPUserInformation.swift in Sources */ = {isa = PBXBuildFile; fileRef = B1217AD643A9E8F88B60F645 /* STPUserInformation.swift */; }; 27F1783CBFEC06BFD6C114F6 /* STPPaymentMethodKlarnaTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 4FD94FF270165D699DA89B24 /* STPPaymentMethodKlarnaTests.swift */; }; 28538CD5885636DC523E8751 /* STPSourceRedirectTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = C0436A8574E7D0730641407A /* STPSourceRedirectTest.swift */; }; @@ -53,7 +49,6 @@ 2AE9ABA774B430E174279FEA /* stp_test_upload_image.jpeg in Resources */ = {isa = PBXBuildFile; fileRef = FD3398E2352CEA0264F20AEA /* stp_test_upload_image.jpeg */; }; 2BD45625F6F665B60C6CAD30 /* STPAddressViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = 28A50AFA4603E488FF3D82D0 /* STPAddressViewModel.swift */; }; 2C6DC246DD12FE0D87156A4D /* STPPaymentMethodPayPalTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 2A63AC868755CB4745E7458E /* STPPaymentMethodPayPalTests.swift */; }; - 2C7991FDF7B374E0E65E253F /* STPPaymentOptionsViewControllerTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = A2922A32A754CFC9AB8B48AE /* STPPaymentOptionsViewControllerTest.swift */; }; 2C9F69E4A384C5743F4EAF69 /* STPPaymentMethodSwishParamsTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 9631915F03F157A1CC3FEFFE /* STPPaymentMethodSwishParamsTests.swift */; }; 2CD7968DA48F7129E16EA0CB /* OHHTTPStubsSwift in Frameworks */ = {isa = PBXBuildFile; productRef = 911CA85A1610303FA0AF0643 /* OHHTTPStubsSwift */; }; 2E35B0FB60FCBE7608080642 /* STPPushProvisioningContext.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6887F19BB9804BF45FD703FF /* STPPushProvisioningContext.swift */; }; @@ -94,7 +89,6 @@ 4059301B0365BD4220E591FB /* STPPaymentMethodSwishTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5DE9F0BAC35EA14579775033 /* STPPaymentMethodSwishTests.swift */; }; 420F8CAB4FAD6D9AF4AF25C0 /* StripePaymentSheet.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = DDC55CC034022DFAC9366E2E /* StripePaymentSheet.framework */; }; 42395EF962DB8AD6A094630B /* StripePaymentSheet.framework in Embed Frameworks */ = {isa = PBXBuildFile; fileRef = DDC55CC034022DFAC9366E2E /* StripePaymentSheet.framework */; settings = {ATTRIBUTES = (CodeSignOnCopy, RemoveHeadersOnCopy, ); }; }; - 429DBA641E926EBC2D049FE7 /* STPCustomerContext.swift in Sources */ = {isa = PBXBuildFile; fileRef = BB8FCDBC63A79CD1571A2DFB /* STPCustomerContext.swift */; }; 42F18560F3DC6980408AF051 /* STPPaymentMethodPayPalParamsTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = DE2A766FB355DD9C461939C1 /* STPPaymentMethodPayPalParamsTests.swift */; }; 43FFF2881D4EFA7B57A60E09 /* STPPaymentMethodCardTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 8BD02D8298877F10F2EF2A9D /* STPPaymentMethodCardTest.swift */; }; 44672917D3AC4B83F9EC3BC3 /* UIView+Stripe_SafeAreaBounds.swift in Sources */ = {isa = PBXBuildFile; fileRef = 86798C95A778362EF815B4C6 /* UIView+Stripe_SafeAreaBounds.swift */; }; @@ -110,7 +104,6 @@ 4A61DC36F10B9C9C24345613 /* STPRadarSessionFunctionalTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 2D878F923A1F69B58D6B2812 /* STPRadarSessionFunctionalTest.swift */; }; 4AAA2CD5AEF1F913395B3B95 /* STPPaymentMethodUSBankAccountParamsStubbedTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = CF13BAEF86594C9CABD4F42A /* STPPaymentMethodUSBankAccountParamsStubbedTest.swift */; }; 4B0917FC15BF56D0100E0ED1 /* STPGenericInputTextFieldSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 58A53F005EA8FDDAA66126BA /* STPGenericInputTextFieldSnapshotTests.swift */; }; - 4C3B161481D11385352B06D4 /* STPCustomerContextTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 2CC4B06AB5C02FF54091E5A8 /* STPCustomerContextTest.swift */; }; 4E09E54E7FEC35C49C59A379 /* STPPushProvisioningDetailsParams.swift in Sources */ = {isa = PBXBuildFile; fileRef = 2B28B8A547CD846277ECD578 /* STPPushProvisioningDetailsParams.swift */; }; 4E31B1864DA407598FB1BBC6 /* STPPostalCodeInputTextFieldTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6618739767139C25C05B3631 /* STPPostalCodeInputTextFieldTests.swift */; }; 4ED44ACF24949F516867235C /* STPPaymentOptionTableViewCell.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6215A9BF343775B1BD0F62AF /* STPPaymentOptionTableViewCell.swift */; }; @@ -139,7 +132,6 @@ 5E498CDA0115CF9F8463C566 /* STPPaymentMethodAUBECSDebitParamsTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7FF2B9FF57E100301B5C38DB /* STPPaymentMethodAUBECSDebitParamsTests.swift */; }; 5E5EE69D140F6FEDA5F0A346 /* STPAPIClientTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = B8DD70E5ED8E9DE8E9752C9E /* STPAPIClientTest.swift */; }; 5ECED204FD22CFEA3A806767 /* STPPaymentOptionTuple.swift in Sources */ = {isa = PBXBuildFile; fileRef = EDD30E5DB8DB3AA3567F5C20 /* STPPaymentOptionTuple.swift */; }; - 605EFBDD21426FD30581563F /* STPAnalyticsClient+BasicUI.swift in Sources */ = {isa = PBXBuildFile; fileRef = 12DBB3F72AEFB52DE27C27ED /* STPAnalyticsClient+BasicUI.swift */; }; 609C2C8F10AFAA2711639CD0 /* NSArray+StripeTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = C17799DC7FA54E758EED31A6 /* NSArray+StripeTest.swift */; }; 609E4D384B75F6A111DC0E27 /* STPPaymentActivityIndicatorView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1FFBAA4B44967B157A4F4E91 /* STPPaymentActivityIndicatorView.swift */; }; 610DF5DC2B33597500DA6AAA /* HostedSurfaceTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 610DF5DB2B33597500DA6AAA /* HostedSurfaceTest.swift */; }; @@ -156,7 +148,6 @@ 66B7EF2DC1CBF813707C767C /* STPBSBNumberValidatorTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = F3C732C25FD961631BD44FDD /* STPBSBNumberValidatorTests.swift */; }; 68318DB86DFCD19505FC47BA /* NSURLComponents_StripeTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = E1CD20E00EAD41091B71ABD5 /* NSURLComponents_StripeTest.swift */; }; 687517E7FE02FFB96DCE2328 /* STPEphemeralKeyManagerTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = C645F78B3EFFAA083B6FD3E9 /* STPEphemeralKeyManagerTest.swift */; }; - 69AC1EDE2A3C03B1D980CA54 /* STPPaymentOptionsViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = A9A1BB31C7B514984231125B /* STPPaymentOptionsViewController.swift */; }; 6BA4B91A2BF433B200D1F21D /* STPPaymentMethodMobilePayParamsTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6BA4B9192BF433B200D1F21D /* STPPaymentMethodMobilePayParamsTests.swift */; }; 6BA4B91C2BF4343B00D1F21D /* STPPaymentMethodMobilePayTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6BA4B91B2BF4343B00D1F21D /* STPPaymentMethodMobilePayTests.swift */; }; 6BC5EC2D2B4609FF00CC75E8 /* LinkInlineSignupElementSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6BC5EC2C2B4609FF00CC75E8 /* LinkInlineSignupElementSnapshotTests.swift */; }; @@ -169,7 +160,6 @@ 6FCA954C32AB351F902BA876 /* STPPostalCodeValidatorTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = FDF7394DDD552EDE996EAD8E /* STPPostalCodeValidatorTest.swift */; }; 701C464523173C6809544935 /* STPThreeDSUICustomizationTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3ED44491EB0AC72B1B1A773C /* STPThreeDSUICustomizationTest.swift */; }; 71116C2D5831E271E12DB059 /* ServerErrorMapperTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 20879436DCFB1F03BE1608B3 /* ServerErrorMapperTest.swift */; }; - 724429607B2741CF44D9C2E5 /* STPShippingMethodsViewControllerLocalizationSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 661976D1296DFA48A25E0493 /* STPShippingMethodsViewControllerLocalizationSnapshotTests.swift */; }; 73AFE2A8839EFAB8330F6CF0 /* STPPaymentIntentTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = ACE8998EAF997A78759E49B5 /* STPPaymentIntentTest.swift */; }; 7435E6BB6971012A9B0DB52E /* STPErrorBridgeTest.m in Sources */ = {isa = PBXBuildFile; fileRef = 9EAE9A2AE65771403CE57C11 /* STPErrorBridgeTest.m */; }; 7589E37795D21AB818B0C333 /* STPAnalyticsClient+Payments.swift in Sources */ = {isa = PBXBuildFile; fileRef = BD89580A3E41D7167C30B287 /* STPAnalyticsClient+Payments.swift */; }; @@ -189,10 +179,8 @@ 7D2C0D1BF455625997CBC33B /* STPAPIClientNetworkBridgeTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = B9BA8D8467218C7E691C9FAE /* STPAPIClientNetworkBridgeTest.swift */; }; 7EAA7334372DBC38DF8FA0AA /* STPPinManagementService.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3EBB07171F6FDCE6E20C454A /* STPPinManagementService.swift */; }; 7F235CD649F6E97E4E7DD180 /* UIView+Stripe_FirstResponder.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1B76DF0FE363F59BF0940A8B /* UIView+Stripe_FirstResponder.swift */; }; - 7F9D08AC5A448C7693162D7D /* STPShippingMethodsViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 21E4B84223DDA131544DBBA7 /* STPShippingMethodsViewController.swift */; }; 801F417CE53689B95C4A098B /* STPBankAccountTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = D103BC590F1E0EC0C31C7B5F /* STPBankAccountTest.swift */; }; 812682EA323986B8F698FF3C /* STPPaymentMethodParams+BasicUI.swift in Sources */ = {isa = PBXBuildFile; fileRef = 53C5AB22D6328E85A6DDF663 /* STPPaymentMethodParams+BasicUI.swift */; }; - 829D43B6705D125FEC9926DA /* STPPaymentContextApplePayTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3C995125252BED1EEC018B9D /* STPPaymentContextApplePayTest.swift */; }; 8378F2A4B0796819BB1C6C54 /* STPPaymentMethodCardParamsTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1671EC46C713D51013AD7D8B /* STPPaymentMethodCardParamsTest.swift */; }; 8520A27C204A068C43592024 /* StripeApplePay.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 52F8AEC50D4623F80F04A533 /* StripeApplePay.framework */; }; 8532FEBF4F2E0EB282D466CE /* STPGenericInputPickerFieldValidatorTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 2D63B73C5773432CA134D1FC /* STPGenericInputPickerFieldValidatorTest.swift */; }; @@ -213,11 +201,9 @@ 951344464ACF84F0F6D43D10 /* OneTimeCodeTextFieldTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 98F9CB667BC68767DFB5FACD /* OneTimeCodeTextFieldTests.swift */; }; 9535CADFFBC9E1FA291E947E /* STPPIIFunctionalTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1645D9793463E266501B74FD /* STPPIIFunctionalTest.swift */; }; 96098727EFA6A72087A35A52 /* STPFormTextFieldTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 2E1862744F23286D1FB9D4AE /* STPFormTextFieldTest.swift */; }; - 97756805F41DDB51B3ED0326 /* STPPaymentContextSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = B5B86F3355E44DF4A980B82C /* STPPaymentContextSnapshotTests.swift */; }; 98E2332DE7F54E970BE5EEF7 /* UIBarButtonItem+Stripe.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3B3000668A75E095B514241F /* UIBarButtonItem+Stripe.swift */; }; 98EE8326C1D133E1C998114F /* STPLocalizedString.swift in Sources */ = {isa = PBXBuildFile; fileRef = AFF957F38AABE5F748C38C0B /* STPLocalizedString.swift */; }; 9A24970C5FB6D3F7314AE550 /* STPAPIClientStubbedTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = E4194E605BB5F31E9CBB8F96 /* STPAPIClientStubbedTest.swift */; }; - 9A57C50938A66604FF16A882 /* STPPaymentOptionsViewControllerLocalizationSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5062915D961C1BCAFD641FFE /* STPPaymentOptionsViewControllerLocalizationSnapshotTests.swift */; }; 9B149DA42FB38C3542E0CB4B /* STPApplePayFunctionalTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3C77C7BC4BA57EC296CF2F1C /* STPApplePayFunctionalTest.swift */; }; 9B1AC278FDCDABF26C5E468C /* STPPostalCodeInputTextFieldSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 090EF7D598B8DE779C275395 /* STPPostalCodeInputTextFieldSnapshotTests.swift */; }; 9C13E8A017A4E23BCCDE618B /* UINavigationController+Stripe_Completion.swift in Sources */ = {isa = PBXBuildFile; fileRef = 2C078573F46762353664AC92 /* UINavigationController+Stripe_Completion.swift */; }; @@ -235,7 +221,6 @@ A77C5769B20D7884FC8FC4FB /* STPNumericDigitInputTextFormatterTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6223E57D3A198F956A37ED89 /* STPNumericDigitInputTextFormatterTests.swift */; }; A781FB0F586B26655FAEC3C0 /* STPCertTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1D51B04D83D4FEF7F90DF16A /* STPCertTest.swift */; }; A8B0DB753CAA2223C8BED099 /* StripeErrorTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = DC3AD586DDED620B9E68F461 /* StripeErrorTest.swift */; }; - A930DF2880EAD0CB9096E49E /* STPShippingAddressViewControllerLocalizationSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1ED6BAC91E8A827DCDB38B15 /* STPShippingAddressViewControllerLocalizationSnapshotTests.swift */; }; AC35943F1EAD50E9D5D509B3 /* STPCardExpiryInputTextFieldFormatterTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 40BB87E28719FE0C6B946BB5 /* STPCardExpiryInputTextFieldFormatterTests.swift */; }; AC7C127B11A60222465F4696 /* XCTest.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 005650A59D692F820EF20F5F /* XCTest.framework */; }; ACC1B91FC687AFD0DFD27CD4 /* STPIntentActionTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 33B9D01D037909D1C9C0B617 /* STPIntentActionTest.swift */; }; @@ -246,7 +231,6 @@ AF18D569B296BFC1EB5A7338 /* ImageTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = AAF368BCD5990EE5DC17D299 /* ImageTest.swift */; }; AF23CB4EF17E87007CFC3E96 /* STPFPXBankStatusResponse.swift in Sources */ = {isa = PBXBuildFile; fileRef = BA08DCDD421CE92ECB61EF5C /* STPFPXBankStatusResponse.swift */; }; AF44725558E654548FED2A2B /* STPPaymentMethodUPITests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 940209E5D30E86E856016906 /* STPPaymentMethodUPITests.swift */; }; - B00F7FC372E376C6B2170D37 /* STPPaymentContext.swift in Sources */ = {isa = PBXBuildFile; fileRef = C750C2C4AB33BC232D1592BA /* STPPaymentContext.swift */; }; B1BF689B91D538BDCA4C8578 /* STPCoreViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 02FC9ED423D40C88D5A24441 /* STPCoreViewController.swift */; }; B359F6DCB31EAD0814AD9AFD /* STPPaymentMethodSEPADebitTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = A41F721AEBB942BB81408A59 /* STPPaymentMethodSEPADebitTest.swift */; }; B44E4CF6C65522F80C946775 /* STPPaymentMethodFunctionalTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 8192839B0F1AE9D9F2A94504 /* STPPaymentMethodFunctionalTest.swift */; }; @@ -263,7 +247,6 @@ B917BF282C84507292112B9D /* STPCardBINMetadataTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3E1C5E08678292561255B1C5 /* STPCardBINMetadataTests.swift */; }; B98D71ED9ACC2E1B47372F53 /* NSDecimalNumber+StripeTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 20552E792B8E7BA15821AB5D /* NSDecimalNumber+StripeTest.swift */; }; BAFD06E994739E1C38DFFBBC /* STPCardScanner.swift in Sources */ = {isa = PBXBuildFile; fileRef = 69BD038947E8E2376A0D240B /* STPCardScanner.swift */; }; - BB46077C256C26418420F240 /* STPAddCardViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = FA793904C7B2D3AA0A4D5EFB /* STPAddCardViewController.swift */; }; BBB734F006FAD749678B87D1 /* STPPaymentMethodRevolutPayParamsTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3AE7BEADD3824A06C2994854 /* STPPaymentMethodRevolutPayParamsTests.swift */; }; BC6912C0DE15008C8D8C303C /* STPFloatingPlaceholderTextFieldSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = F7385193226663A5B79E69ED /* STPFloatingPlaceholderTextFieldSnapshotTests.swift */; }; BC694A1642DC30D530B60635 /* RotatingCardBrandsViewSnapshotTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = FDA32D0C9E8A7A69F4899EDC /* RotatingCardBrandsViewSnapshotTests.swift */; }; @@ -298,7 +281,6 @@ CBAF9C6F87F746F17495ADC2 /* STPPaymentMethodCashAppTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7DDE50CBC86AD77084C877B6 /* STPPaymentMethodCashAppTests.swift */; }; CBCA59D39B30D869B4FDC04B /* STPE2ETest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1C1548BA518F7AC2A9ECF9D5 /* STPE2ETest.swift */; }; CC072EBAD035AA54A2AD3ABC /* UIViewController+Stripe_KeyboardAvoiding.swift in Sources */ = {isa = PBXBuildFile; fileRef = 2F08757CA6F6B2DA65C14E0A /* UIViewController+Stripe_KeyboardAvoiding.swift */; }; - CEE483EB7B06B3C607BC755C /* UINavigationBar+StripeTest.m in Sources */ = {isa = PBXBuildFile; fileRef = E68F6B90F3BC61A49570FAF4 /* UINavigationBar+StripeTest.m */; }; CEF318C74D2E44C78EF85306 /* STPBankSelectionTableViewCell.swift in Sources */ = {isa = PBXBuildFile; fileRef = 458F8576215E0F8ECE1D74CE /* STPBankSelectionTableViewCell.swift */; }; CF2E17AC77EB08393B8A3F98 /* STPCardNumberInputTextFieldFormatterTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = D3803D0DED98501AA26B2EAC /* STPCardNumberInputTextFieldFormatterTests.swift */; }; CFC1F2B8D48FFF7B0F81B5A0 /* STPPaymentMethodAddressTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = BEB3F9F0228008BE213706DF /* STPPaymentMethodAddressTest.swift */; }; @@ -364,7 +346,6 @@ F835CEC935464FF32726A0A0 /* STPTheme.swift in Sources */ = {isa = PBXBuildFile; fileRef = 8E8CA4361964E1BA400EFC89 /* STPTheme.swift */; }; F86F2DF6E46EFABE23AD5D27 /* STPApplePayContextDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = E452877E5D11120B1E28A6E7 /* STPApplePayContextDelegate.swift */; }; F975CE029DF30419B8DB0D8F /* STPNumericStringValidatorTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 4AA36705DED9164663A98B6A /* STPNumericStringValidatorTests.swift */; }; - FB34906C9215D0E03850064B /* STPAddCardViewControllerTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = 8704ABFA91A5226847F4A69A /* STPAddCardViewControllerTest.swift */; }; FBBA3B39598BBECB664C5E7F /* STPApplePayTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = CFC4BC1AB047ED88C4D13C89 /* STPApplePayTest.swift */; }; FC166455478EAF51F7C34E68 /* STPApplePayPaymentOption.swift in Sources */ = {isa = PBXBuildFile; fileRef = 03ACDC7EEC28D1FE50008F65 /* STPApplePayPaymentOption.swift */; }; FDADE3E36804A8AD82301BF3 /* STPPaymentMethodAfterpayClearpayParamsTest.swift in Sources */ = {isa = PBXBuildFile; fileRef = ECBA5B09A3FD875C93218573 /* STPPaymentMethodAfterpayClearpayParamsTest.swift */; }; @@ -483,7 +464,6 @@ 121B7EDCCD0957C9A444A8E3 /* ru */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = ru; path = ru.lproj/Localizable.strings; sourceTree = ""; }; 12632E6710DE8861CAF1BAA4 /* RotatingCardBrandsViewTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = RotatingCardBrandsViewTests.swift; sourceTree = ""; }; 12D757B36030685C401A6990 /* et-EE */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "et-EE"; path = "et-EE.lproj/Localizable.strings"; sourceTree = ""; }; - 12DBB3F72AEFB52DE27C27ED /* STPAnalyticsClient+BasicUI.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "STPAnalyticsClient+BasicUI.swift"; sourceTree = ""; }; 13DDBEA7D444A8AC14E0F1C8 /* lv-LV */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "lv-LV"; path = "lv-LV.lproj/Localizable.strings"; sourceTree = ""; }; 148C1D7D1BBBC6B74894A869 /* STPPaymentMethodTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodTest.swift; sourceTree = ""; }; 153071C69A0BEE033E035DCF /* CardExpiryDateTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CardExpiryDateTests.swift; sourceTree = ""; }; @@ -504,7 +484,6 @@ 1DD6897858F46976A946394E /* StripeiOSAppHostedTests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = StripeiOSAppHostedTests.xctest; sourceTree = BUILT_PRODUCTS_DIR; }; 1E2638F7AA0906914117C2D5 /* STPPaymentMethodOptionsTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodOptionsTest.swift; sourceTree = ""; }; 1E8AFAE24610EC983727F860 /* STPAPIClient+BasicUI.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "STPAPIClient+BasicUI.swift"; sourceTree = ""; }; - 1ED6BAC91E8A827DCDB38B15 /* STPShippingAddressViewControllerLocalizationSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPShippingAddressViewControllerLocalizationSnapshotTests.swift; sourceTree = ""; }; 1F0DF2ED9232A7CC51F5FCB1 /* STPPaymentMethodAffirmTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodAffirmTests.swift; sourceTree = ""; }; 1F16C36797D978E72E612100 /* STPPaymentCardTextFieldTestsSwift.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentCardTextFieldTestsSwift.swift; sourceTree = ""; }; 1F29C15B47C7CB0941CD4C9E /* ViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ViewController.swift; sourceTree = ""; }; @@ -514,12 +493,10 @@ 207677E2A0DBC04C88139372 /* STPPaymentMethodSofortParamsTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodSofortParamsTests.swift; sourceTree = ""; }; 20879436DCFB1F03BE1608B3 /* ServerErrorMapperTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ServerErrorMapperTest.swift; sourceTree = ""; }; 21780C410D22264B7C299520 /* STPSourceSEPADebitDetailsTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPSourceSEPADebitDetailsTest.swift; sourceTree = ""; }; - 21E4B84223DDA131544DBBA7 /* STPShippingMethodsViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPShippingMethodsViewController.swift; sourceTree = ""; }; 22D1C6EB5826E2D7C80B6CF3 /* StripePayments.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = StripePayments.framework; sourceTree = BUILT_PRODUCTS_DIR; }; 23997D61DF41CA84BFC33080 /* STPBECSDebitAccountNumberValidatorTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPBECSDebitAccountNumberValidatorTests.swift; sourceTree = ""; }; 2560B4EEE60D40FB31B8552F /* de */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = de; path = de.lproj/Localizable.strings; sourceTree = ""; }; 26302721419496F37DE91DF8 /* UserDefaults+Stripe.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "UserDefaults+Stripe.swift"; sourceTree = ""; }; - 26E5C1DABAA63F07F0C6AE37 /* STPAddCardViewControllerLocalizationSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPAddCardViewControllerLocalizationSnapshotTests.swift; sourceTree = ""; }; 26E70469F4032553B4BB62DA /* ca-ES */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "ca-ES"; path = "ca-ES.lproj/Localizable.strings"; sourceTree = ""; }; 273EE407039913F0B644172B /* PKAddPaymentPassRequest+Stripe_Error.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "PKAddPaymentPassRequest+Stripe_Error.swift"; sourceTree = ""; }; 27A4D83C0E7D4AE0CCD38B89 /* da */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = da; path = da.lproj/Localizable.strings; sourceTree = ""; }; @@ -530,7 +507,6 @@ 2A8A2CD759D465290066EF65 /* STPTextFieldDelegateProxyTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPTextFieldDelegateProxyTests.swift; sourceTree = ""; }; 2B28B8A547CD846277ECD578 /* STPPushProvisioningDetailsParams.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPushProvisioningDetailsParams.swift; sourceTree = ""; }; 2C078573F46762353664AC92 /* UINavigationController+Stripe_Completion.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "UINavigationController+Stripe_Completion.swift"; sourceTree = ""; }; - 2CC4B06AB5C02FF54091E5A8 /* STPCustomerContextTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCustomerContextTest.swift; sourceTree = ""; }; 2D1525AF65BDEF691F8BCBE8 /* STPCoreScrollViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCoreScrollViewController.swift; sourceTree = ""; }; 2D63B73C5773432CA134D1FC /* STPGenericInputPickerFieldValidatorTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPGenericInputPickerFieldValidatorTest.swift; sourceTree = ""; }; 2D878F923A1F69B58D6B2812 /* STPRadarSessionFunctionalTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPRadarSessionFunctionalTest.swift; sourceTree = ""; }; @@ -553,7 +529,6 @@ 3B3000668A75E095B514241F /* UIBarButtonItem+Stripe.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "UIBarButtonItem+Stripe.swift"; sourceTree = ""; }; 3C742844915B96CFD25BFFF9 /* AfterpayPriceBreakdownViewSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AfterpayPriceBreakdownViewSnapshotTests.swift; sourceTree = ""; }; 3C77C7BC4BA57EC296CF2F1C /* STPApplePayFunctionalTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPApplePayFunctionalTest.swift; sourceTree = ""; }; - 3C995125252BED1EEC018B9D /* STPPaymentContextApplePayTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentContextApplePayTest.swift; sourceTree = ""; }; 3CDD1E823223F450193E8746 /* STPPaymentMethodAfterpayClearpayTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodAfterpayClearpayTest.swift; sourceTree = ""; }; 3E1C5E08678292561255B1C5 /* STPCardBINMetadataTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCardBINMetadataTests.swift; sourceTree = ""; }; 3E5FB20B2BEFC00D54FDD87D /* STPCard+BasicUI.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "STPCard+BasicUI.swift"; sourceTree = ""; }; @@ -586,7 +561,6 @@ 4E371E9B3B2E343FE954531C /* STPPaymentMethodBoletoTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodBoletoTests.swift; sourceTree = ""; }; 4FD94FF270165D699DA89B24 /* STPPaymentMethodKlarnaTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodKlarnaTests.swift; sourceTree = ""; }; 4FFA8B446217CDE678D7287F /* STPBlocks.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = STPBlocks.h; sourceTree = ""; }; - 5062915D961C1BCAFD641FFE /* STPPaymentOptionsViewControllerLocalizationSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentOptionsViewControllerLocalizationSnapshotTests.swift; sourceTree = ""; }; 512A0E7C246D5F044245E069 /* StripeCoreTestUtils.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = StripeCoreTestUtils.framework; sourceTree = BUILT_PRODUCTS_DIR; }; 51408DE266D0345784ADD4FA /* STPThreeDSNavigationBarCustomizationTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPThreeDSNavigationBarCustomizationTest.swift; sourceTree = ""; }; 51BD2CE41E4F0CF648F44E4A /* TextFieldElement+IBANTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "TextFieldElement+IBANTest.swift"; sourceTree = ""; }; @@ -625,7 +599,6 @@ 655209238C85F466F9F14F14 /* STPPaymentMethodBancontactParamsTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodBancontactParamsTests.swift; sourceTree = ""; }; 65DCC9BDA647E58D3882C698 /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist; path = Info.plist; sourceTree = ""; }; 6618739767139C25C05B3631 /* STPPostalCodeInputTextFieldTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPostalCodeInputTextFieldTests.swift; sourceTree = ""; }; - 661976D1296DFA48A25E0493 /* STPShippingMethodsViewControllerLocalizationSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPShippingMethodsViewControllerLocalizationSnapshotTests.swift; sourceTree = ""; }; 683F7735569D22CBEC9CA2E6 /* STPPaymentHandlerFunctionalTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentHandlerFunctionalTest.m; sourceTree = ""; }; 6887F19BB9804BF45FD703FF /* STPPushProvisioningContext.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPushProvisioningContext.swift; sourceTree = ""; }; 6955B3A3353F8442E4FBBBF6 /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Assets.xcassets; sourceTree = ""; }; @@ -640,13 +613,11 @@ 6E1F6514E7530C2A3478B2F5 /* STPCardCVCInputTextFieldFormatterTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCardCVCInputTextFieldFormatterTests.swift; sourceTree = ""; }; 6F01150CD0255164FE2CF3A4 /* STPPaymentIntentParams+BasicUI.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "STPPaymentIntentParams+BasicUI.swift"; sourceTree = ""; }; 6F017A08C7E633FB4297D274 /* UIViewController+Stripe_NavigationItemProxy.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "UIViewController+Stripe_NavigationItemProxy.swift"; sourceTree = ""; }; - 71711FC8E2FB66E52A5FDD9A /* STPShippingAddressViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPShippingAddressViewController.swift; sourceTree = ""; }; 72903593DC432D01720DC9D9 /* STPAddressFieldTableViewCell.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPAddressFieldTableViewCell.swift; sourceTree = ""; }; 739934737B9A09775CD278C9 /* STPPaymentMethodNetBankingTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodNetBankingTests.swift; sourceTree = ""; }; 74FDEF9F687C63BADFB96480 /* STPPaymentMethodUSBankAccountParamsTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodUSBankAccountParamsTest.swift; sourceTree = ""; }; 77247622AB08FEF48CA0DC26 /* StripePaymentsTestUtils.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = StripePaymentsTestUtils.framework; sourceTree = BUILT_PRODUCTS_DIR; }; 77E846CD56018D8417A3AB95 /* StripeiOS.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = StripeiOS.xcassets; sourceTree = ""; }; - 789D0B49B0788794739E3DD4 /* STPShippingAddressViewControllerTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPShippingAddressViewControllerTest.swift; sourceTree = ""; }; 78D1EEABBAE5BD5615486B0F /* STPLabeledFormTextFieldViewSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPLabeledFormTextFieldViewSnapshotTests.swift; sourceTree = ""; }; 79ABE6A14AF9D14103050876 /* STPPaymentConfigurationTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentConfigurationTest.m; sourceTree = ""; }; 7A517686D4D12691351311CA /* STPIntentActionPayNowDisplayQrCodeTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPIntentActionPayNowDisplayQrCodeTest.swift; sourceTree = ""; }; @@ -673,7 +644,6 @@ 85C29AE44D809CD677B5E52B /* STPFPXBankBrandTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPFPXBankBrandTest.swift; sourceTree = ""; }; 86798C95A778362EF815B4C6 /* UIView+Stripe_SafeAreaBounds.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "UIView+Stripe_SafeAreaBounds.swift"; sourceTree = ""; }; 86983B09A1712944EC012AD4 /* STPViewWithSeparatorSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPViewWithSeparatorSnapshotTests.swift; sourceTree = ""; }; - 8704ABFA91A5226847F4A69A /* STPAddCardViewControllerTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPAddCardViewControllerTest.swift; sourceTree = ""; }; 88639E3C3AC622B5EF475538 /* STPUIVCStripeParentViewControllerTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPUIVCStripeParentViewControllerTests.swift; sourceTree = ""; }; 88AEABC15CCBB9EA393C175F /* STPMocks.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPMocks.m; sourceTree = ""; }; 890660C21E3666CE7B82695B /* STPEphemeralKey.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPEphemeralKey.swift; sourceTree = ""; }; @@ -717,7 +687,6 @@ A1C67AC5D415615E9F27D3E3 /* STPPaymentMethodBoletoParamsTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodBoletoParamsTests.swift; sourceTree = ""; }; A1C876DC7F3E31D7189506A8 /* STPPaymentContextAmountModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentContextAmountModel.swift; sourceTree = ""; }; A22E5B87755C1F05C3DB438C /* STPInputTextFieldFormatterTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPInputTextFieldFormatterTests.swift; sourceTree = ""; }; - A2922A32A754CFC9AB8B48AE /* STPPaymentOptionsViewControllerTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentOptionsViewControllerTest.swift; sourceTree = ""; }; A39580123A4F1EA96F91768A /* STPAddressTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPAddressTests.swift; sourceTree = ""; }; A41F721AEBB942BB81408A59 /* STPPaymentMethodSEPADebitTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodSEPADebitTest.swift; sourceTree = ""; }; A5398E1156E0BFEBBF56FD2F /* String+Localized.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "String+Localized.swift"; sourceTree = ""; }; @@ -727,7 +696,6 @@ A6F6634AD12771A9BB100DD3 /* es */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = es; path = es.lproj/Localizable.strings; sourceTree = ""; }; A7E369CEC9F5B3758F78E88F /* UINavigationBar+Stripe_Theme.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "UINavigationBar+Stripe_Theme.swift"; sourceTree = ""; }; A8598727045C6268B57A5FC7 /* Stripe-umbrella.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "Stripe-umbrella.h"; sourceTree = ""; }; - A9A1BB31C7B514984231125B /* STPPaymentOptionsViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentOptionsViewController.swift; sourceTree = ""; }; AA3CA5B4B91838CC7ED4D5EB /* ro-RO */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "ro-RO"; path = "ro-RO.lproj/Localizable.strings"; sourceTree = ""; }; AAF368BCD5990EE5DC17D299 /* ImageTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ImageTest.swift; sourceTree = ""; }; AB1A92C77AC61335184BBDBC /* es-419 */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = "es-419"; path = "es-419.lproj/Localizable.strings"; sourceTree = ""; }; @@ -744,7 +712,6 @@ B407FE2D39775902A95B1118 /* StripeiOS Tests-Bridging-Header.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "StripeiOS Tests-Bridging-Header.h"; sourceTree = ""; }; B4D12508C2F1056A7EAFEC86 /* PKPayment+StripeTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "PKPayment+StripeTest.swift"; sourceTree = ""; }; B4D31B0D7BD9F97AF3BB61E6 /* StripeBundleLocator.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = StripeBundleLocator.swift; sourceTree = ""; }; - B5B86F3355E44DF4A980B82C /* STPPaymentContextSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentContextSnapshotTests.swift; sourceTree = ""; }; B669C53A601CD3CB0203A4B9 /* STPAPISettingsObjCBridgeTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPAPISettingsObjCBridgeTest.m; sourceTree = ""; }; B6EC2F562B618A3D00FF72A2 /* STPBasicUIAnalyticsSerializer.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPBasicUIAnalyticsSerializer.swift; sourceTree = ""; }; B6F3B966470A530E0DC53F8C /* STPThreeDSButtonCustomizationTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPThreeDSButtonCustomizationTest.swift; sourceTree = ""; }; @@ -756,7 +723,6 @@ BA08DCDD421CE92ECB61EF5C /* STPFPXBankStatusResponse.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPFPXBankStatusResponse.swift; sourceTree = ""; }; BAFD938F3A149A56CC99FE96 /* fi */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = fi; path = fi.lproj/Localizable.strings; sourceTree = ""; }; BB08D2AC882B21C8ADD76B92 /* STPPaymentCardTextFieldKVOTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = STPPaymentCardTextFieldKVOTest.m; sourceTree = ""; }; - BB8FCDBC63A79CD1571A2DFB /* STPCustomerContext.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCustomerContext.swift; sourceTree = ""; }; BCBEA9E4823F08C1F5057B5A /* STPAUBECSDebitFormViewSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPAUBECSDebitFormViewSnapshotTests.swift; sourceTree = ""; }; BD77D4B1C5B64E45F9DA09B5 /* STPConfirmCardOptionsTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPConfirmCardOptionsTest.swift; sourceTree = ""; }; BD89580A3E41D7167C30B287 /* STPAnalyticsClient+Payments.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "STPAnalyticsClient+Payments.swift"; sourceTree = ""; }; @@ -773,7 +739,6 @@ C641A744CCEA67C07E9BFE05 /* NSLocale+STPSwizzling.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "NSLocale+STPSwizzling.swift"; sourceTree = ""; }; C645F78B3EFFAA083B6FD3E9 /* STPEphemeralKeyManagerTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPEphemeralKeyManagerTest.swift; sourceTree = ""; }; C713F58BC61A962C720AE0AE /* STPMandateCustomerAcceptanceParamsTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPMandateCustomerAcceptanceParamsTest.swift; sourceTree = ""; }; - C750C2C4AB33BC232D1592BA /* STPPaymentContext.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentContext.swift; sourceTree = ""; }; C8F8FCC84601E4ADC6B7F3CE /* PaymentAnalyticTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PaymentAnalyticTest.swift; sourceTree = ""; }; C980D24DDC884FECCE39139F /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.storyboard; name = Base; path = Base.lproj/Main.storyboard; sourceTree = ""; }; CA8B8F540CD05B3DC2C5EEA6 /* STPSetupIntentTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPSetupIntentTest.swift; sourceTree = ""; }; @@ -798,7 +763,6 @@ D3E0CA28591EB0748C64D1FA /* STPFileTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPFileTest.swift; sourceTree = ""; }; D41081066DC4465734F7FCD7 /* STPPaymentMethodNetBankingParamsTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodNetBankingParamsTest.swift; sourceTree = ""; }; D42F83F785EAF24F5DC7ED1A /* STPCardCVCInputTextFieldValidatorTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCardCVCInputTextFieldValidatorTests.swift; sourceTree = ""; }; - D5C4A4CC7D2E9B5AB3EC3B79 /* STPPaymentOptionsInternalViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentOptionsInternalViewController.swift; sourceTree = ""; }; D609FFD051FC01BF566665A0 /* StripeiOS Tests-Debug.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = "StripeiOS Tests-Debug.xcconfig"; sourceTree = ""; }; D794C5E6396B4A19DC4F6921 /* StripeUICore.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = StripeUICore.framework; sourceTree = BUILT_PRODUCTS_DIR; }; D7C4773C2D193BEDF1CBB530 /* STPCardNumberInputTextFieldValidatorTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCardNumberInputTextFieldValidatorTests.swift; sourceTree = ""; }; @@ -826,7 +790,6 @@ E452877E5D11120B1E28A6E7 /* STPApplePayContextDelegate.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPApplePayContextDelegate.swift; sourceTree = ""; }; E5D9F97ABC88302478220267 /* PKPaymentAuthorizationViewController+Stripe_Blocks.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "PKPaymentAuthorizationViewController+Stripe_Blocks.swift"; sourceTree = ""; }; E6312182B5BCAB940D216650 /* ConsumerSessionTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ConsumerSessionTests.swift; sourceTree = ""; }; - E68F6B90F3BC61A49570FAF4 /* UINavigationBar+StripeTest.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = "UINavigationBar+StripeTest.m"; sourceTree = ""; }; E6DEE912364C9F4B51B374D0 /* STPPaymentCardTextFieldViewModelTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentCardTextFieldViewModelTest.swift; sourceTree = ""; }; E7ACB4FAFAD33296DE34D036 /* STPAnalyticsClientPaymentsTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPAnalyticsClientPaymentsTest.swift; sourceTree = ""; }; E7C7D85A7FAAFDF4F59BA85E /* LinkSignupViewModelTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = LinkSignupViewModelTests.swift; sourceTree = ""; }; @@ -852,7 +815,6 @@ F546088BA4F763334CFD3D34 /* STPImageLibraryTest.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPImageLibraryTest.swift; sourceTree = ""; }; F6558C62376C2397030BD4A6 /* STPPaymentMethodPrzelewy24Tests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentMethodPrzelewy24Tests.swift; sourceTree = ""; }; F7385193226663A5B79E69ED /* STPFloatingPlaceholderTextFieldSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPFloatingPlaceholderTextFieldSnapshotTests.swift; sourceTree = ""; }; - FA793904C7B2D3AA0A4D5EFB /* STPAddCardViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPAddCardViewController.swift; sourceTree = ""; }; FC30E6129279F14506219E98 /* STPPaymentHandlerTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPPaymentHandlerTests.swift; sourceTree = ""; }; FD289E1EA9F0CE1C848AC0BB /* STPCardNumberInputTextFieldSnapshotTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = STPCardNumberInputTextFieldSnapshotTests.swift; sourceTree = ""; }; FD3398E2352CEA0264F20AEA /* stp_test_upload_image.jpeg */ = {isa = PBXFileReference; lastKnownFileType = image.jpeg; path = stp_test_upload_image.jpeg; sourceTree = ""; }; @@ -947,11 +909,9 @@ 8F7E3CE2105E4A39032CD919 /* Enums+CustomStringConvertible.swift */, 273EE407039913F0B644172B /* PKAddPaymentPassRequest+Stripe_Error.swift */, E5D9F97ABC88302478220267 /* PKPaymentAuthorizationViewController+Stripe_Blocks.swift */, - FA793904C7B2D3AA0A4D5EFB /* STPAddCardViewController.swift */, 498C3FB07CFD532779C755D3 /* STPAddress+BasicUI.swift */, 72903593DC432D01720DC9D9 /* STPAddressFieldTableViewCell.swift */, 28A50AFA4603E488FF3D82D0 /* STPAddressViewModel.swift */, - 12DBB3F72AEFB52DE27C27ED /* STPAnalyticsClient+BasicUI.swift */, B6EC2F562B618A3D00FF72A2 /* STPBasicUIAnalyticsSerializer.swift */, BD89580A3E41D7167C30B287 /* STPAnalyticsClient+Payments.swift */, 1E8AFAE24610EC983727F860 /* STPAPIClient+BasicUI.swift */, @@ -970,7 +930,6 @@ 2D1525AF65BDEF691F8BCBE8 /* STPCoreScrollViewController.swift */, B78C72B0DB434EC7F700FDE0 /* STPCoreTableViewController.swift */, 02FC9ED423D40C88D5A24441 /* STPCoreViewController.swift */, - BB8FCDBC63A79CD1571A2DFB /* STPCustomerContext.swift */, 890660C21E3666CE7B82695B /* STPEphemeralKey.swift */, 588C260880FFC584A00A89F5 /* STPEphemeralKeyManager.swift */, 485E747DA1F72F091986787B /* STPEphemeralKeyProvider.swift */, @@ -982,14 +941,11 @@ 1FFBAA4B44967B157A4F4E91 /* STPPaymentActivityIndicatorView.swift */, 01B42BE6FB5EC1F708875AB8 /* STPPaymentCardTextFieldCell.swift */, 18FCB69CD3B8C3DAB216A5F0 /* STPPaymentConfiguration.swift */, - C750C2C4AB33BC232D1592BA /* STPPaymentContext.swift */, A1C876DC7F3E31D7189506A8 /* STPPaymentContextAmountModel.swift */, 6F01150CD0255164FE2CF3A4 /* STPPaymentIntentParams+BasicUI.swift */, 35C1E9B0EE03825DABF6471A /* STPPaymentMethod+BasicUI.swift */, 53C5AB22D6328E85A6DDF663 /* STPPaymentMethodParams+BasicUI.swift */, 30B694A39D54886392AA5DE3 /* STPPaymentOption.swift */, - D5C4A4CC7D2E9B5AB3EC3B79 /* STPPaymentOptionsInternalViewController.swift */, - A9A1BB31C7B514984231125B /* STPPaymentOptionsViewController.swift */, 6215A9BF343775B1BD0F62AF /* STPPaymentOptionTableViewCell.swift */, EDD30E5DB8DB3AA3567F5C20 /* STPPaymentOptionTuple.swift */, 7D964D9E01627B419B4BD23C /* STPPaymentResult.swift */, @@ -998,8 +954,6 @@ 89E5DA3029F141B5111A5B2C /* STPPushProvisioningDetails.swift */, 2B28B8A547CD846277ECD578 /* STPPushProvisioningDetailsParams.swift */, 96E1FED5CE5974C9C1162E93 /* STPSectionHeaderView.swift */, - 71711FC8E2FB66E52A5FDD9A /* STPShippingAddressViewController.swift */, - 21E4B84223DDA131544DBBA7 /* STPShippingMethodsViewController.swift */, 98544B08552407D41D398C68 /* STPShippingMethodTableViewCell.swift */, C3B75875C55D2C2723DC5090 /* STPSource+BasicUI.swift */, 8E8CA4361964E1BA400EFC89 /* STPTheme.swift */, @@ -1164,8 +1118,6 @@ FDA32D0C9E8A7A69F4899EDC /* RotatingCardBrandsViewSnapshotTests.swift */, 12632E6710DE8861CAF1BAA4 /* RotatingCardBrandsViewTests.swift */, 20879436DCFB1F03BE1608B3 /* ServerErrorMapperTest.swift */, - 26E5C1DABAA63F07F0C6AE37 /* STPAddCardViewControllerLocalizationSnapshotTests.swift */, - 8704ABFA91A5226847F4A69A /* STPAddCardViewControllerTest.swift */, A39580123A4F1EA96F91768A /* STPAddressTests.swift */, D3D2DB08C335695B705F544C /* STPAddressViewModelTest.swift */, E7ACB4FAFAD33296DE34D036 /* STPAnalyticsClientPaymentsTest.swift */, @@ -1214,7 +1166,6 @@ E97018A201D01A8FA59999C2 /* STPConnectAccountFunctionalTest.swift */, 195DEC752CC82CC4BA1E2351 /* STPConnectAccountParamsTest.swift */, 8D49257A97E71A475A9F6E08 /* STPCountryPickerInputFieldSnapshotTests.swift */, - 2CC4B06AB5C02FF54091E5A8 /* STPCustomerContextTest.swift */, DAB817ED5B5DB87AE1290894 /* STPCustomerTest.swift */, 1C1548BA518F7AC2A9ECF9D5 /* STPE2ETest.swift */, C645F78B3EFFAA083B6FD3E9 /* STPEphemeralKeyManagerTest.swift */, @@ -1254,8 +1205,6 @@ 1F16C36797D978E72E612100 /* STPPaymentCardTextFieldTestsSwift.swift */, E6DEE912364C9F4B51B374D0 /* STPPaymentCardTextFieldViewModelTest.swift */, 79ABE6A14AF9D14103050876 /* STPPaymentConfigurationTest.m */, - 3C995125252BED1EEC018B9D /* STPPaymentContextApplePayTest.swift */, - B5B86F3355E44DF4A980B82C /* STPPaymentContextSnapshotTests.swift */, 683F7735569D22CBEC9CA2E6 /* STPPaymentHandlerFunctionalTest.m */, ACF450AD17FF7BCE5916DDF1 /* STPPaymentHandlerFunctionalTest.swift */, EF48EC440E1ED5D6BAA567FF /* STPPaymentHandlerStubbedMockedFilesTests.swift */, @@ -1330,8 +1279,6 @@ CF13BAEF86594C9CABD4F42A /* STPPaymentMethodUSBankAccountParamsStubbedTest.swift */, 74FDEF9F687C63BADFB96480 /* STPPaymentMethodUSBankAccountParamsTest.swift */, 3B112FFF3FCA82094281493F /* STPPaymentMethodUSBankAccountTest.swift */, - 5062915D961C1BCAFD641FFE /* STPPaymentOptionsViewControllerLocalizationSnapshotTests.swift */, - A2922A32A754CFC9AB8B48AE /* STPPaymentOptionsViewControllerTest.swift */, 924E878428D15506711CA628 /* STPPhoneNumberValidatorTest.swift */, 1645D9793463E266501B74FD /* STPPIIFunctionalTest.swift */, 336CC555B845DED30208D39D /* STPPinManagementServiceFunctionalTest.swift */, @@ -1347,9 +1294,6 @@ 49AA313E068FB99CEAA5F7D3 /* STPSetupIntentFunctionalTest.swift */, 01B057D99A14E5BA6019C349 /* STPSetupIntentLastSetupErrorTest.swift */, CA8B8F540CD05B3DC2C5EEA6 /* STPSetupIntentTest.swift */, - 1ED6BAC91E8A827DCDB38B15 /* STPShippingAddressViewControllerLocalizationSnapshotTests.swift */, - 789D0B49B0788794739E3DD4 /* STPShippingAddressViewControllerTest.swift */, - 661976D1296DFA48A25E0493 /* STPShippingMethodsViewControllerLocalizationSnapshotTests.swift */, 63114D0EAAE2606732DF5AA0 /* STPSourceCardDetailsTest.swift */, 17013F78CE3F9662029FEF5B /* STPSourceFunctionalTest.swift */, 095EBF095BA2BC8D299547DB /* STPSourceOwnerTest.swift */, @@ -1376,7 +1320,6 @@ DC3AD586DDED620B9E68F461 /* StripeErrorTest.swift */, B407FE2D39775902A95B1118 /* StripeiOS Tests-Bridging-Header.h */, 51BD2CE41E4F0CF648F44E4A /* TextFieldElement+IBANTest.swift */, - E68F6B90F3BC61A49570FAF4 /* UINavigationBar+StripeTest.m */, 3B0E131538728BC4802627B1 /* UserDefaults+StripeTest.swift */, 0DB03E83746FE78361831546 /* WalletHeaderViewSnapshotTests.swift */, CAC3A0332C2F176A007BC888 /* STPPaymentMethodSunbitTests.swift */, @@ -1630,12 +1573,10 @@ C0688E067AE4FFDFFDDC03BB /* PKPaymentAuthorizationViewController+Stripe_Blocks.swift in Sources */, 6BF6ECC4A4E61E2FFC3EA20B /* STPAPIClient+BasicUI.swift in Sources */, 5212C7875C07F9BF16AFD98D /* STPAPIClient+PushProvisioning.swift in Sources */, - BB46077C256C26418420F240 /* STPAddCardViewController.swift in Sources */, E63B5BAF6B5645C979BFBA71 /* STPAddress+BasicUI.swift in Sources */, 58240AA66FAE55131268E4A0 /* STPAddressFieldTableViewCell.swift in Sources */, 313F5F792B0BE59100BD98A9 /* Docs.docc in Sources */, 2BD45625F6F665B60C6CAD30 /* STPAddressViewModel.swift in Sources */, - 605EFBDD21426FD30581563F /* STPAnalyticsClient+BasicUI.swift in Sources */, 7589E37795D21AB818B0C333 /* STPAnalyticsClient+Payments.swift in Sources */, F86F2DF6E46EFABE23AD5D27 /* STPApplePayContextDelegate.swift in Sources */, FC166455478EAF51F7C34E68 /* STPApplePayPaymentOption.swift in Sources */, @@ -1651,7 +1592,6 @@ DF73457BF349BC962A6AC502 /* STPCoreScrollViewController.swift in Sources */, EEFFE199D9769FF449BFD7FF /* STPCoreTableViewController.swift in Sources */, B1BF689B91D538BDCA4C8578 /* STPCoreViewController.swift in Sources */, - 429DBA641E926EBC2D049FE7 /* STPCustomerContext.swift in Sources */, 62B91808A088C4F9FDB62C53 /* STPEphemeralKey.swift in Sources */, 4FB67F10A0B7106A8142B842 /* STPEphemeralKeyManager.swift in Sources */, B8385576DC25BDEEB92D812F /* STPEphemeralKeyProvider.swift in Sources */, @@ -1664,7 +1604,6 @@ 609E4D384B75F6A111DC0E27 /* STPPaymentActivityIndicatorView.swift in Sources */, 4EFF8B46B12DA4D9AAB22523 /* STPPaymentCardTextFieldCell.swift in Sources */, 446A108C8EB6C338A1D774F8 /* STPPaymentConfiguration.swift in Sources */, - B00F7FC372E376C6B2170D37 /* STPPaymentContext.swift in Sources */, 447C19BDB2CF5445045F81F7 /* STPPaymentContextAmountModel.swift in Sources */, EA7FEC518AA07BA59405A5E3 /* STPPaymentIntentParams+BasicUI.swift in Sources */, 7BC98BE168781C5B3EC8A8DB /* STPPaymentMethod+BasicUI.swift in Sources */, @@ -1672,17 +1611,13 @@ F10FC337254A34ED8F13E341 /* STPPaymentOption.swift in Sources */, 4ED44ACF24949F516867235C /* STPPaymentOptionTableViewCell.swift in Sources */, 5ECED204FD22CFEA3A806767 /* STPPaymentOptionTuple.swift in Sources */, - 1F432D0B37949217E4299A20 /* STPPaymentOptionsInternalViewController.swift in Sources */, - 69AC1EDE2A3C03B1D980CA54 /* STPPaymentOptionsViewController.swift in Sources */, 307FD6A103EF7AF3CE451598 /* STPPaymentResult.swift in Sources */, 7EAA7334372DBC38DF8FA0AA /* STPPinManagementService.swift in Sources */, 2E35B0FB60FCBE7608080642 /* STPPushProvisioningContext.swift in Sources */, FEE74744B657F86873EA2F3D /* STPPushProvisioningDetails.swift in Sources */, 4E09E54E7FEC35C49C59A379 /* STPPushProvisioningDetailsParams.swift in Sources */, 23D1246A5DAB5333650F104F /* STPSectionHeaderView.swift in Sources */, - 124D43C1A633922B1DA3E1E7 /* STPShippingAddressViewController.swift in Sources */, 7B9C0D039EA9EF593AEC682D /* STPShippingMethodTableViewCell.swift in Sources */, - 7F9D08AC5A448C7693162D7D /* STPShippingMethodsViewController.swift in Sources */, 3CE88568CB9648D6F1503B88 /* STPSource+BasicUI.swift in Sources */, F835CEC935464FF32726A0A0 /* STPTheme.swift in Sources */, 279D2BA91198E18730626CE6 /* STPUserInformation.swift in Sources */, @@ -1766,8 +1701,6 @@ AD09F2ACD0CDCBD414AC30AD /* STPAPISettingsObjCBridgeTest.m in Sources */, 17BD7C0391F3182E32A63D6B /* STPAUBECSDebitFormViewSnapshotTests.swift in Sources */, 583DE9869C885BA02E0A071E /* STPAUBECSFormViewModelTests.swift in Sources */, - 26F38A2A57FDDC12926BE044 /* STPAddCardViewControllerLocalizationSnapshotTests.swift in Sources */, - FB34906C9215D0E03850064B /* STPAddCardViewControllerTest.swift in Sources */, 3EA9D509E59DA65EE4EDF98D /* STPAddressTests.swift in Sources */, F53E04785DB804EA5C2AAC18 /* STPAddressViewModelTest.swift in Sources */, CAC3A0342C2F176A007BC888 /* STPPaymentMethodSunbitTests.swift in Sources */, @@ -1811,7 +1744,6 @@ 14656D177E67594B8C75A9FE /* STPConnectAccountParamsTest.swift in Sources */, 610DF5DC2B33597500DA6AAA /* HostedSurfaceTest.swift in Sources */, 9D464A252FBD0D4E2A0A7398 /* STPCountryPickerInputFieldSnapshotTests.swift in Sources */, - 4C3B161481D11385352B06D4 /* STPCustomerContextTest.swift in Sources */, 6EDFC83541EED9E361B71C02 /* STPCustomerTest.swift in Sources */, CBCA59D39B30D869B4FDC04B /* STPE2ETest.swift in Sources */, 687517E7FE02FFB96DCE2328 /* STPEphemeralKeyManagerTest.swift in Sources */, @@ -1854,8 +1786,6 @@ B4719234E4BBDAD260E31373 /* STPPaymentCardTextFieldViewModelTest.swift in Sources */, 3AAE488F2461A46143B3A687 /* STPPaymentConfigurationTest.m in Sources */, CAC80EBF2C33339D001E3D0D /* STPPaymentMethodSatispayTests.swift in Sources */, - 829D43B6705D125FEC9926DA /* STPPaymentContextApplePayTest.swift in Sources */, - 97756805F41DDB51B3ED0326 /* STPPaymentContextSnapshotTests.swift in Sources */, FEF2E0DAC862FF42B814AFCA /* STPPaymentHandlerFunctionalTest.m in Sources */, 8F5AF9D3566B8DBCA5AB5188 /* STPPaymentHandlerFunctionalTest.swift in Sources */, 194154708E1A9E013DCE2C72 /* STPPaymentHandlerStubbedMockedFilesTests.swift in Sources */, @@ -1923,8 +1853,6 @@ 7D251ABF1EBF65ACA8A4BDD4 /* STPPaymentMethodUSBankAccountParamsTest.swift in Sources */, 225140E0BD9C0630116DDE4A /* STPPaymentMethodUSBankAccountTest.swift in Sources */, B71F04D02538FA1723558C48 /* STPPaymentMethodiDEALTest.swift in Sources */, - 9A57C50938A66604FF16A882 /* STPPaymentOptionsViewControllerLocalizationSnapshotTests.swift in Sources */, - 2C7991FDF7B374E0E65E253F /* STPPaymentOptionsViewControllerTest.swift in Sources */, 07BF3CF1656AF5F5A0678873 /* STPPhoneNumberValidatorTest.swift in Sources */, EBD436689635CC28A24DECD4 /* STPPinManagementServiceFunctionalTest.swift in Sources */, 385CAC4D2FF119D2E925916B /* STPPostalCodeInputTextFieldFormatterTests.swift in Sources */, @@ -1940,9 +1868,6 @@ C32D7ACEBC852CBC295BBEF2 /* STPSetupIntentFunctionalTest.swift in Sources */, E699508F4DB4D9D4666BAA08 /* STPSetupIntentLastSetupErrorTest.swift in Sources */, EC4DC8E386544959E1AA9355 /* STPSetupIntentTest.swift in Sources */, - A930DF2880EAD0CB9096E49E /* STPShippingAddressViewControllerLocalizationSnapshotTests.swift in Sources */, - 08ED7A4EB7E64FDAED2C2D39 /* STPShippingAddressViewControllerTest.swift in Sources */, - 724429607B2741CF44D9C2E5 /* STPShippingMethodsViewControllerLocalizationSnapshotTests.swift in Sources */, AE747ADA2841AA06F32558D8 /* STPSourceCardDetailsTest.swift in Sources */, 3FD5ABC45AF3A03F4EFE196F /* STPSourceFunctionalTest.swift in Sources */, 922C0DF37F5AAA29375A5454 /* STPSourceOwnerTest.swift in Sources */, @@ -1971,7 +1896,6 @@ 71116C2D5831E271E12DB059 /* ServerErrorMapperTest.swift in Sources */, A8B0DB753CAA2223C8BED099 /* StripeErrorTest.swift in Sources */, 3B237145902E3DB07E747E32 /* TextFieldElement+IBANTest.swift in Sources */, - CEE483EB7B06B3C607BC755C /* UINavigationBar+StripeTest.m in Sources */, 51D515315F02D4C03BA12366 /* UserDefaults+StripeTest.swift in Sources */, 8B80FB6FC88D411A90E9D487 /* WalletHeaderViewSnapshotTests.swift in Sources */, ); diff --git a/Stripe/StripeiOS/Source/STPAddCardViewController.swift b/Stripe/StripeiOS/Source/STPAddCardViewController.swift deleted file mode 100644 index 853908fd4ba..00000000000 --- a/Stripe/StripeiOS/Source/STPAddCardViewController.swift +++ /dev/null @@ -1,1001 +0,0 @@ -// -// STPAddCardViewController.swift -// StripeiOS -// -// Created by Jack Flintermann on 3/23/16. -// Copyright © 2016 Stripe, Inc. All rights reserved. -// - -@_spi(STP) import StripeCore -@_spi(STP) import StripePayments -@_spi(STP) import StripePaymentsUI -@_spi(STP) import StripeUICore -import UIKit - -/// This view controller contains a credit card entry form that the user can fill out. On submission, it will use the Stripe API to convert the user's card details to a Stripe token. It renders a right bar button item that submits the form, so it must be shown inside a `UINavigationController`. -public class STPAddCardViewController: STPCoreTableViewController, STPAddressViewModelDelegate, - STPCardScannerDelegate, STPPaymentCardTextFieldDelegate, UITableViewDelegate, - UITableViewDataSource -{ - - /// A convenience initializer; equivalent to calling `init(configuration: STPPaymentConfiguration.shared, theme: STPTheme.defaultTheme)`. - @objc - public convenience init() { - self.init(configuration: STPPaymentConfiguration.shared, theme: STPTheme.defaultTheme) - } - - /// Initializes a new `STPAddCardViewController` with the provided configuration and theme. Don't forget to set the `delegate` property after initialization. - /// - Parameters: - /// - configuration: The configuration to use (this determines the Stripe publishable key to use, the required billing address fields, whether or not to use SMS autofill, etc). - seealso: STPPaymentConfiguration - /// - theme: The theme to use to inform the view controller's visual appearance. - seealso: STPTheme - @objc(initWithConfiguration:theme:) - public init( - configuration: STPPaymentConfiguration, - theme: STPTheme - ) { - addressViewModel = STPAddressViewModel( - requiredBillingFields: configuration.requiredBillingAddressFields, - availableCountries: configuration.availableCountries - ) - super.init(theme: theme) - commonInit(with: configuration) - } - - /// The view controller's delegate. This must be set before showing the view controller in order for it to work properly. - seealso: STPAddCardViewControllerDelegate - @objc public weak var delegate: STPAddCardViewControllerDelegate? - /// You can set this property to pre-fill any information you've already collected from your user. - seealso: STPUserInformation.h - @objc public var prefilledInformation: STPUserInformation? { - didSet { - if let address = prefilledInformation?.billingAddress { - addressViewModel.address = address - } - } - } - // Should be overwritten if this class is used by STPPaymentContext or STPPaymentOptionsViewController - internal var analyticsLogger: STPPaymentContext.AnalyticsLogger = .init(product: STPAddCardViewController.self) - private var _customFooterView: UIView? - /// Provide this view controller with a footer view. - /// When the footer view needs to be resized, it will be sent a - /// `sizeThatFits:` call. The view should respond correctly to this method in order - /// to be sized and positioned properly. - @objc public var customFooterView: UIView? { - get { - _customFooterView - } - set(footerView) { - _customFooterView = footerView - _configureFooterView() - } - } - - func _configureFooterView() { - if isViewLoaded, let footerView = _customFooterView { - let size = footerView.sizeThatFits( - CGSize(width: view.bounds.size.width, height: CGFloat.greatestFiniteMagnitude) - ) - footerView.frame = CGRect(x: 0, y: 0, width: size.width, height: size.height) - - tableView?.tableFooterView = footerView - } - } - - /// The API Client to use to make requests. - /// Defaults to `STPAPIClient.shared` - public var apiClient: STPAPIClient = STPAPIClient.shared - - /// Use init: or initWithConfiguration:theme: - required init( - theme: STPTheme? - ) { - let configuration = STPPaymentConfiguration.shared - addressViewModel = STPAddressViewModel( - requiredBillingFields: configuration.requiredBillingAddressFields, - availableCountries: configuration.availableCountries - ) - super.init(theme: theme) - } - - /// Use init: or initWithConfiguration:theme: - required init( - nibName nibNameOrNil: String?, - bundle nibBundleOrNil: Bundle? - ) { - let configuration = STPPaymentConfiguration.shared - addressViewModel = STPAddressViewModel( - requiredBillingFields: configuration.requiredBillingAddressFields, - availableCountries: configuration.availableCountries - ) - super.init(nibName: nibNameOrNil, bundle: nibBundleOrNil) - } - - /// Use init: or initWithConfiguration:theme: - required init?( - coder aDecoder: NSCoder - ) { - let configuration = STPPaymentConfiguration.shared - addressViewModel = STPAddressViewModel( - requiredBillingFields: configuration.requiredBillingAddressFields, - availableCountries: configuration.availableCountries - ) - super.init(coder: aDecoder) - } - - private var _alwaysEnableDoneButton = false - @objc var alwaysEnableDoneButton: Bool { - get { - _alwaysEnableDoneButton - } - set(alwaysEnableDoneButton) { - if alwaysEnableDoneButton != _alwaysEnableDoneButton { - _alwaysEnableDoneButton = alwaysEnableDoneButton - updateDoneButton() - } - } - } - private var configuration: STPPaymentConfiguration? - @objc var shippingAddress: STPAddress? - private var hasUsedShippingAddress = false - private weak var cardImageView: UIImageView? - private var doneItem: UIBarButtonItem? - private var cardHeaderView: STPSectionHeaderView? - - @available(macCatalyst 14.0, *) - private var cardScanner: STPCardScanner? { - get { - _cardScanner as? STPCardScanner - } - set { - _cardScanner = newValue - } - } - - /// Storage for `cardScanner`. - private var _cardScanner: NSObject? - - @available(macCatalyst 14.0, *) - private var scannerCell: STPCardScannerTableViewCell? { - get { - _scannerCell as? STPCardScannerTableViewCell - } - set { - _scannerCell = newValue - } - } - - /// Storage for `scannerCell`. - private var _scannerCell: NSObject? - - private var _isScanning = false - private var isScanning: Bool { - get { - _isScanning - } - set(isScanning) { - if _isScanning == isScanning { - return - } - _isScanning = isScanning - - cardHeaderView?.button?.isEnabled = !isScanning - let indexPath = IndexPath( - row: 0, - section: STPPaymentCardSection.stpPaymentCardScannerSection.rawValue - ) - tableView?.beginUpdates() - if isScanning { - tableView?.insertRows(at: [indexPath], with: .automatic) - } else { - tableView?.deleteRows(at: [indexPath], with: .automatic) - } - tableView?.endUpdates() - if isScanning { - tableView?.scrollToRow(at: indexPath, at: .middle, animated: true) - } - updateInputAccessoryVisiblity() - } - } - private var addressHeaderView: STPSectionHeaderView? - var paymentCell: STPPaymentCardTextFieldCell? - var viewDidAppearStartTime: Date? - - private var _loading = false - @objc var loading: Bool { - get { - _loading - } - set(loading) { - if loading == _loading { - return - } - _loading = loading - stp_navigationItemProxy?.setHidesBackButton(loading, animated: true) - stp_navigationItemProxy?.leftBarButtonItem?.isEnabled = !loading - activityIndicator?.animating = loading - if loading { - tableView?.endEditing(true) - var loadingItem: UIBarButtonItem? - if let activityIndicator = activityIndicator { - loadingItem = UIBarButtonItem(customView: activityIndicator) - } - stp_navigationItemProxy?.setRightBarButton(loadingItem, animated: true) - cardHeaderView?.buttonHidden = true - } else { - stp_navigationItemProxy?.setRightBarButton(doneItem, animated: true) - cardHeaderView?.buttonHidden = false - } - var cells = addressViewModel.addressCells as [UITableViewCell] - - if let paymentCell = paymentCell { - cells.append(paymentCell) - } - for cell in cells { - cell.isUserInteractionEnabled = !loading - UIView.animate( - withDuration: 0.1, - animations: { - cell.alpha = loading ? 0.7 : 1.0 - } - ) - } - } - } - private var activityIndicator: STPPaymentActivityIndicatorView? - private weak var lookupActivityIndicator: STPPaymentActivityIndicatorView? - var addressViewModel: STPAddressViewModel - private var inputAccessoryToolbar: UIToolbar? - private var lookupSucceeded = false - private var scannerCompleteAnimationTimer: Timer? - var didSendFormInteractedAnalytic = false - - @objc(commonInitWithConfiguration:) func commonInit(with configuration: STPPaymentConfiguration) - { - STPAnalyticsClient.sharedClient.addClass( - toProductUsageIfNecessary: STPAddCardViewController.self - ) - - self.configuration = configuration - shippingAddress = nil - hasUsedShippingAddress = false - addressViewModel.delegate = self - title = STPLocalizedString("Add a Card", "Title for Add a Card view") - - if #available(macCatalyst 14.0, *) { - cardScanner = STPCardScanner() - } - } - - /// :nodoc: - @objc - public func tableView( - _ tableView: UITableView, - estimatedHeightForRowAt indexPath: IndexPath - ) -> CGFloat { - return 44.0 - } - - @objc override func createAndSetupViews() { - super.createAndSetupViews() - - let doneItem = UIBarButtonItem( - barButtonSystemItem: .done, - target: self, - action: #selector(nextPressed(_:)) - ) - self.doneItem = doneItem - stp_navigationItemProxy?.rightBarButtonItem = doneItem - updateDoneButton() - - stp_navigationItemProxy?.leftBarButtonItem?.accessibilityIdentifier = - "AddCardViewControllerNavBarCancelButtonIdentifier" - stp_navigationItemProxy?.rightBarButtonItem?.accessibilityIdentifier = - "AddCardViewControllerNavBarDoneButtonIdentifier" - - let cardImageView = UIImageView(image: STPLegacyImageLibrary.largeCardFrontImage()) - cardImageView.contentMode = .center - cardImageView.frame = CGRect( - x: 0, - y: 0, - width: view.bounds.size.width, - height: cardImageView.bounds.size.height + (57 * 2) - ) - self.cardImageView = cardImageView - tableView?.tableHeaderView = cardImageView - - let paymentCell = STPPaymentCardTextFieldCell( - style: .default, - reuseIdentifier: "STPAddCardViewControllerPaymentCardTextFieldCell" - ) - paymentCell.paymentField?.delegate = self - if configuration?.requiredBillingAddressFields == .postalCode { - // If postal code collection is enabled, move the postal code field into the card entry field. - // Otherwise, this will be picked up by the billing address fields below. - paymentCell.paymentField?.postalCodeEntryEnabled = true - } - self.paymentCell = paymentCell - - activityIndicator = STPPaymentActivityIndicatorView( - frame: CGRect(x: 0, y: 0, width: 20.0, height: 20.0) - ) - - inputAccessoryToolbar = UIToolbar.stp_inputAccessoryToolbar( - withTarget: self, - action: #selector(paymentFieldNextTapped) - ) - inputAccessoryToolbar?.stp_setEnabled(false) - updateInputAccessoryVisiblity() - tableView?.dataSource = self - tableView?.delegate = self - tableView?.reloadData() - if let address = prefilledInformation?.billingAddress { - addressViewModel.address = address - } - - let addressHeaderView = STPSectionHeaderView() - addressHeaderView.theme = theme - addressHeaderView.title = String.Localized.billing_address - switch configuration?.shippingType { - case .shipping: - addressHeaderView.button?.setTitle( - STPLocalizedString( - "Use Shipping", - "Button to fill billing address from shipping address." - ), - for: .normal - ) - case .delivery: - addressHeaderView.button?.setTitle( - STPLocalizedString( - "Use Delivery", - "Button to fill billing address from delivery address." - ), - for: .normal - ) - default: - break - } - addressHeaderView.button?.addTarget( - self, - action: #selector(useShippingAddress(_:)), - for: .touchUpInside - ) - let requiredFields = configuration?.requiredBillingAddressFields ?? .none - let needsAddress = requiredFields != .none && !addressViewModel.isValid - let buttonVisible = - needsAddress && shippingAddress?.containsContent(for: requiredFields) != nil - && !hasUsedShippingAddress - addressHeaderView.buttonHidden = !buttonVisible - addressHeaderView.setNeedsLayout() - self.addressHeaderView = addressHeaderView - let cardHeaderView = STPSectionHeaderView() - cardHeaderView.theme = theme - cardHeaderView.title = STPPaymentMethodType.card.displayName - cardHeaderView.buttonHidden = true - self.cardHeaderView = cardHeaderView - - // re-set the custom footer view if it was added before we loaded - _configureFooterView() - - view.addGestureRecognizer( - UITapGestureRecognizer(target: self, action: #selector(endEditing)) - ) - - setUpCardScanningIfAvailable() - - STPAnalyticsClient.sharedClient.clearAdditionalInfo() - } - - /// :nodoc: - @objc - public override func viewDidLayoutSubviews() { - super.viewDidLayoutSubviews() - - // Resetting it re-calculates the size based on new view width - // UITableView requires us to call setter again to actually pick up frame - // change on footers - if tableView?.tableFooterView != nil { - customFooterView = tableView?.tableFooterView - } - } - - func setUpCardScanningIfAvailable() { - if #available(macCatalyst 14.0, *) { - if !STPCardScanner.cardScanningAvailable || configuration?.cardScanningEnabled != true { - return - } - let scannerCell = STPCardScannerTableViewCell() - self.scannerCell = scannerCell - - let cardScanner = STPCardScanner(delegate: self) - cardScanner.cameraView = scannerCell.cameraView - self.cardScanner = cardScanner - - cardHeaderView?.buttonHidden = false - cardHeaderView?.button?.setTitle( - String.Localized.scan_card_title_capitalization, - for: .normal - ) - cardHeaderView?.button?.addTarget( - self, - action: #selector(scanCard), - for: .touchUpInside - ) - cardHeaderView?.setNeedsLayout() - } - } - - @available(macCatalyst 14.0, *) - @objc func scanCard() { - view.endEditing(true) - isScanning = true - cardScanner?.start() - sendFormInteractedAnalyticIfNecessary() - } - - @objc func endEditing() { - view.endEditing(false) - } - - /// :nodoc: - @objc - public override func updateAppearance() { - super.updateAppearance() - - view.backgroundColor = theme.primaryBackgroundColor - - let navBarTheme = navigationController?.navigationBar.stp_theme ?? theme - doneItem?.stp_setTheme(navBarTheme) - tableView?.allowsSelection = false - - cardImageView?.tintColor = theme.accentColor - activityIndicator?.tintColor = theme.accentColor - - paymentCell?.theme = theme - cardHeaderView?.theme = theme - addressHeaderView?.theme = theme - for cell in addressViewModel.addressCells { - cell.theme = theme - } - setNeedsStatusBarAppearanceUpdate() - } - - /// :nodoc: - @objc - public override func viewDidAppear(_ animated: Bool) { - super.viewDidAppear(animated) - viewDidAppearStartTime = Date() - analyticsLogger.logFormShown(paymentMethodType: .card) - - stp_beginObservingKeyboardAndInsettingScrollView( - tableView, - onChange: nil - ) - firstEmptyField()?.becomeFirstResponder() - } - - func firstEmptyField() -> UIResponder? { - - if paymentCell?.isEmpty != nil { - return paymentCell! - } - - for cell in addressViewModel.addressCells { - if cell.contents?.count ?? 0 == 0 { - return cell - } - } - return nil - } - - /// :nodoc: - @objc - public override func handleCancelTapped(_ sender: Any?) { - delegate?.addCardViewControllerDidCancel(self) - } - - @objc func nextPressed(_ sender: Any?) { - analyticsLogger.logDoneButtonTapped(paymentMethodType: .card, shownStartDate: viewDidAppearStartTime) - loading = true - guard let cardParams = paymentCell?.paymentField?.paymentMethodParams.card else { - return - } - // Create and return a Payment Method - let billingDetails = STPPaymentMethodBillingDetails() - if configuration?.requiredBillingAddressFields == .postalCode { - let address = STPAddress() - address.postalCode = paymentCell?.paymentField?.postalCode - billingDetails.address = STPPaymentMethodAddress(address: address) - } else { - billingDetails.address = STPPaymentMethodAddress(address: addressViewModel.address) - billingDetails.email = addressViewModel.address.email - billingDetails.name = addressViewModel.address.name - billingDetails.phone = addressViewModel.address.phone - } - let paymentMethodParams = STPPaymentMethodParams( - card: cardParams, - billingDetails: billingDetails, - metadata: nil - ) - apiClient.createPaymentMethod(with: paymentMethodParams) { - paymentMethod, - createPaymentMethodError in - if let createPaymentMethodError = createPaymentMethodError { - self.handleError(createPaymentMethodError) - } else { - if let paymentMethod = paymentMethod { - self.delegate?.addCardViewController( - self, - didCreatePaymentMethod: paymentMethod - ) { - attachToCustomerError in - stpDispatchToMainThreadIfNecessary({ - if let attachToCustomerError = attachToCustomerError { - self.handleError(attachToCustomerError) - } else { - self.loading = false - } - }) - } - } - } - } - } - - func handleError(_ error: Error) { - loading = false - firstEmptyField()?.becomeFirstResponder() - - let alertController = UIAlertController( - title: error.localizedDescription, - message: (error as NSError).localizedFailureReason, - preferredStyle: .alert - ) - - alertController.addAction( - UIAlertAction( - title: String.Localized.ok, - style: .cancel, - handler: nil - ) - ) - - present(alertController, animated: true) - } - - func updateDoneButton() { - stp_navigationItemProxy?.rightBarButtonItem?.isEnabled = - (paymentCell?.paymentField?.isValid ?? false && addressViewModel.isValid) - || alwaysEnableDoneButton - } - - func updateInputAccessoryVisiblity() { - // The inputAccessoryToolbar switches from the paymentCell to the first address field. - // It should only be shown when there *is* an address field. This compensates for the lack - // of a 'Return' key on the number pad used for paymentCell entry - let hasAddressCells = (addressViewModel.addressCells.count) > 0 - paymentCell?.inputAccessoryView = hasAddressCells ? inputAccessoryToolbar : nil - } - - /// Only send form interacted analytic once per time this screen is shown - func sendFormInteractedAnalyticIfNecessary() { - if !didSendFormInteractedAnalytic { - didSendFormInteractedAnalytic = true - analyticsLogger.logFormInteracted(paymentMethodType: .card) - } - } - - /// Only send card number completed analytic once per time the card number changes from invalid to valid - var shouldSendCardNumberCompletedAnalytic = true - func sendCardNumberCompletedAnalyticIfNecessary(cardNumber: String?) { - let isCardNumberValid = STPCardValidator.validationState(forNumber: cardNumber, validatingCardBrand: true) == .valid - if isCardNumberValid, shouldSendCardNumberCompletedAnalytic { - analyticsLogger.logCardNumberCompleted() - shouldSendCardNumberCompletedAnalytic = false - } else if !isCardNumberValid { - // Reset shouldSendCardNumberCompletedAnalytic when the card number is invalid, so that it gets sent when the card number becomes valid. - shouldSendCardNumberCompletedAnalytic = true - } - } - - // MARK: - STPPaymentCardTextField - @objc - public func paymentCardTextFieldDidChange(_ textField: STPPaymentCardTextField) { - sendFormInteractedAnalyticIfNecessary() - sendCardNumberCompletedAnalyticIfNecessary(cardNumber: textField.cardNumber) - inputAccessoryToolbar?.stp_setEnabled(textField.isValid) - updateDoneButton() - } - - @objc func paymentFieldNextTapped() { - _ = addressViewModel.addressCells.stp_boundSafeObject(at: 0)?.becomeFirstResponder() - } - - @objc - public func paymentCardTextFieldWillEndEditing(forReturn textField: STPPaymentCardTextField) { - paymentFieldNextTapped() - } - - @objc - public func paymentCardTextFieldDidBeginEditingCVC(_ textField: STPPaymentCardTextField) { - let isAmex = STPCardValidator.brand(forNumber: textField.cardNumber ?? "") == .amex - var newImage: UIImage? - var animationTransition: UIView.AnimationOptions - - if isAmex { - newImage = STPLegacyImageLibrary.largeCardAmexCVCImage() - animationTransition = .transitionCrossDissolve - } else { - newImage = STPLegacyImageLibrary.largeCardBackImage() - animationTransition = .transitionFlipFromRight - } - - if let cardImageView = cardImageView { - UIView.transition( - with: cardImageView, - duration: 0.2, - options: animationTransition, - animations: { - self.cardImageView?.image = newImage - } - ) - } - } - - @objc - public func paymentCardTextFieldDidEndEditingCVC(_ textField: STPPaymentCardTextField) { - let isAmex = STPCardValidator.brand(forNumber: textField.cardNumber ?? "") == .amex - let animationTransition: UIView.AnimationOptions = - isAmex ? .transitionCrossDissolve : .transitionFlipFromLeft - - if let cardImageView = cardImageView { - UIView.transition( - with: cardImageView, - duration: 0.2, - options: animationTransition, - animations: { - self.cardImageView?.image = STPLegacyImageLibrary.largeCardFrontImage() - } - ) - } - } - - @objc - public func paymentCardTextFieldDidBeginEditing(_ textField: STPPaymentCardTextField) { - if #available(macCatalyst 14.0, *) { - cardScanner?.stop() - } - } - - // MARK: - STPAddressViewModelDelegate - func addressViewModel(_ addressViewModel: STPAddressViewModel, addedCellAt index: Int) { - let indexPath = IndexPath( - row: index, - section: STPPaymentCardSection.stpPaymentCardBillingAddressSection.rawValue - ) - tableView?.insertRows(at: [indexPath], with: .automatic) - updateInputAccessoryVisiblity() - } - - func addressViewModel(_ addressViewModel: STPAddressViewModel, removedCellAt index: Int) { - let indexPath = IndexPath( - row: Int(index), - section: STPPaymentCardSection.stpPaymentCardBillingAddressSection.rawValue - ) - tableView?.deleteRows(at: [indexPath], with: .automatic) - updateInputAccessoryVisiblity() - } - - func addressViewModelDidChange(_ addressViewModel: STPAddressViewModel) { - updateDoneButton() - } - - func addressViewModelWillUpdate(_ addressViewModel: STPAddressViewModel) { - tableView?.beginUpdates() - } - - func addressViewModelDidUpdate(_ addressViewModel: STPAddressViewModel) { - tableView?.endUpdates() - } - - // MARK: - UITableView - /// :nodoc: - @objc - public func numberOfSections(in tableView: UITableView) -> Int { - return 3 - } - - /// :nodoc: - @objc - public func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int { - if section == STPPaymentCardSection.stpPaymentCardNumberSection.rawValue { - return 1 - } else if section == STPPaymentCardSection.stpPaymentCardScannerSection.rawValue { - return isScanning ? 1 : 0 - } else if section == STPPaymentCardSection.stpPaymentCardBillingAddressSection.rawValue { - return addressViewModel.addressCells.count - } - return 0 - } - - /// :nodoc: - @objc - public func tableView( - _ tableView: UITableView, - cellForRowAt indexPath: IndexPath - ) -> UITableViewCell { - var cell: UITableViewCell? - switch indexPath.section { - case STPPaymentCardSection.stpPaymentCardNumberSection.rawValue: - cell = paymentCell - case STPPaymentCardSection.stpPaymentCardScannerSection.rawValue: - if #available(macCatalyst 14.0, *) { - cell = scannerCell - } else { - return UITableViewCell() - } - case STPPaymentCardSection.stpPaymentCardBillingAddressSection.rawValue: - cell = addressViewModel.addressCells.stp_boundSafeObject(at: indexPath.row) - default: - return UITableViewCell() // won't be called; exists to make the static analyzer happy - } - cell?.backgroundColor = theme.secondaryBackgroundColor - cell?.contentView.backgroundColor = UIColor.clear - return cell! - } - - /// :nodoc: - @objc - public func tableView( - _ tableView: UITableView, - willDisplay cell: UITableViewCell, - forRowAt indexPath: IndexPath - ) { - let topRow = indexPath.row == 0 - let bottomRow = - self.tableView(tableView, numberOfRowsInSection: indexPath.section) - 1 == indexPath.row - cell.stp_setBorderColor(theme.tertiaryBackgroundColor) - cell.stp_setTopBorderHidden(!topRow) - cell.stp_setBottomBorderHidden(!bottomRow) - cell.stp_setFakeSeparatorColor(theme.quaternaryBackgroundColor) - cell.stp_setFakeSeparatorLeftInset(15.0) - } - - /// :nodoc: - @objc - public func tableView( - _ tableView: UITableView, - heightForFooterInSection section: Int - ) - -> CGFloat - { - if self.tableView(tableView, numberOfRowsInSection: section) == 0 { - return 0.01 - } - return 27.0 - } - - /// :nodoc: - @objc - public override func tableView( - _ tableView: UITableView, - heightForHeaderInSection section: Int - ) -> CGFloat { - let fittingSize = CGSize( - width: view.bounds.size.width, - height: CGFloat.greatestFiniteMagnitude - ) - let numberOfRows = self.tableView(tableView, numberOfRowsInSection: section) - if section == STPPaymentCardSection.stpPaymentCardNumberSection.rawValue { - return cardHeaderView?.sizeThatFits(fittingSize).height ?? 0.0 - } else if section == STPPaymentCardSection.stpPaymentCardBillingAddressSection.rawValue - && numberOfRows != 0 - { - return addressHeaderView?.sizeThatFits(fittingSize).height ?? 0.0 - } else if section == STPPaymentCardSection.stpPaymentCardScannerSection.rawValue { - return 0.01 - } else if numberOfRows != 0 { - return tableView.sectionHeaderHeight - } - return 0.01 - } - - /// :nodoc: - @objc - public func tableView( - _ tableView: UITableView, - viewForHeaderInSection section: Int - ) - -> UIView? - { - if self.tableView(tableView, numberOfRowsInSection: section) == 0 { - return UIView() - } else { - if section == STPPaymentCardSection.stpPaymentCardNumberSection.rawValue { - return cardHeaderView - } else if section == STPPaymentCardSection.stpPaymentCardBillingAddressSection.rawValue - { - return addressHeaderView - } - } - return nil - } - - /// :nodoc: - @objc - public func tableView( - _ tableView: UITableView, - viewForFooterInSection section: Int - ) - -> UIView? - { - return UIView() - } - - @objc func useShippingAddress(_ sender: UIButton) { - tableView?.beginUpdates() - addressViewModel.address = shippingAddress ?? STPAddress() - hasUsedShippingAddress = true - firstEmptyField()?.becomeFirstResponder() - UIView.animate( - withDuration: 0.2, - animations: { - self.addressHeaderView?.buttonHidden = true - } - ) - tableView?.endUpdates() - } - - // MARK: - STPCardScanner - /// :nodoc: - @objc - public override func viewWillTransition( - to size: CGSize, - with coordinator: UIViewControllerTransitionCoordinator - ) { - super.viewWillTransition(to: size, with: coordinator) - let orientation = UIDevice.current.orientation - if orientation.isPortrait || orientation.isLandscape { - if #available(macCatalyst 14.0, *) { - cardScanner?.deviceOrientation = orientation - } - } - if isScanning { - let indexPath = IndexPath( - row: 0, - section: STPPaymentCardSection.stpPaymentCardScannerSection.rawValue - ) - DispatchQueue.main.async(execute: { - self.tableView?.scrollToRow(at: indexPath, at: .middle, animated: true) - }) - } - } - - static let cardScannerKSTPCardScanAnimationTime: TimeInterval = 0.04 - - @available(macCatalyst 14.0, *) - func cardScanner( - _ scanner: STPCardScanner, - didFinishWith cardParams: STPPaymentMethodCardParams?, - error: Error? - ) { - if let error = error { - handleError(error) - } - if let cardParams = cardParams { - view.isUserInteractionEnabled = false - paymentCell?.paymentField?.inputView = UIView() as? UIInputView - var i = 0 - scannerCompleteAnimationTimer = Timer.scheduledTimer( - withTimeInterval: STPAddCardViewController.cardScannerKSTPCardScanAnimationTime, - repeats: true, - block: { timer in - i += 1 - let newParams = STPPaymentMethodCardParams() - guard let number = cardParams.number else { - timer.invalidate() - self.view.isUserInteractionEnabled = false - return - } - if i < number.count { - newParams.number = String( - number[...number.index(number.startIndex, offsetBy: i)] - ) - } else { - newParams.number = number - } - self.paymentCell?.paymentField?.paymentMethodParams = STPPaymentMethodParams( - card: newParams, - billingDetails: nil, - metadata: nil - ) - if i > number.count { - self.paymentCell?.paymentField?.paymentMethodParams = - STPPaymentMethodParams( - card: cardParams, - billingDetails: nil, - metadata: nil - ) - self.isScanning = false - self.paymentCell?.paymentField?.inputView = nil - // Force the inputView to reload by asking the text field to resign/become first responder: - _ = self.paymentCell?.paymentField?.resignFirstResponder() - _ = self.paymentCell?.paymentField?.becomeFirstResponder() - timer.invalidate() - self.view.isUserInteractionEnabled = true - } - } - ) - } else { - isScanning = false - } - } - -} - -/// An `STPAddCardViewControllerDelegate` is notified when an `STPAddCardViewController` -/// successfully creates a card token or is cancelled. It has internal error-handling -/// logic, so there's no error case to deal with. -@objc public protocol STPAddCardViewControllerDelegate: NSObjectProtocol { - /// Called when the user cancels adding a card. You should dismiss (or pop) the - /// view controller at this point. - /// - Parameter addCardViewController: the view controller that has been cancelled - func addCardViewControllerDidCancel(_ addCardViewController: STPAddCardViewController) - - /// This is called when the user successfully adds a card and Stripe returns a - /// Payment Method. - /// You should send the PaymentMethod to your backend to store it on a customer, and then - /// call the provided `completion` block when that call is finished. If an error - /// occurs while talking to your backend, call `completion(error)`, otherwise, - /// dismiss (or pop) the view controller. - /// - Parameters: - /// - addCardViewController: the view controller that successfully created a token - /// - paymentMethod: the Payment Method that was created. - seealso: STPPaymentMethod - /// - completion: call this callback when you're done sending the token to your backend - @objc func addCardViewController( - _ addCardViewController: STPAddCardViewController, - didCreatePaymentMethod paymentMethod: STPPaymentMethod, - completion: @escaping STPErrorBlock - ) - - // MARK: - Deprecated - - /// This method is deprecated as of v16.0.0 (https://github.com/stripe/stripe-ios/blob/master/MIGRATING.md#migrating-from-versions--1600). - /// To use this class, migrate your integration from Charges to PaymentIntents. See https://stripe.com/docs/payments/payment-intents/migration/charges#read - @available( - *, - deprecated, - message: - "Use addCardViewController(_:didCreatePaymentMethod:completion:) instead and migrate your integration to PaymentIntents. See https://stripe.com/docs/payments/payment-intents/migration/charges#read", - renamed: "addCardViewController(_:didCreatePaymentMethod:completion:)" - ) - @objc optional func addCardViewController( - _ addCardViewController: STPAddCardViewController, - didCreateToken token: STPToken, - completion: STPErrorBlock - ) - /// This method is deprecated as of v16.0.0 (https://github.com/stripe/stripe-ios/blob/master/MIGRATING.md#migrating-from-versions--1600). - /// To use this class, migrate your integration from Charges to PaymentIntents. See https://stripe.com/docs/payments/payment-intents/migration/charges#read - @available( - *, - deprecated, - message: - "Use addCardViewController(_:didCreatePaymentMethod:completion:) instead and migrate your integration to PaymentIntents. See https://stripe.com/docs/payments/payment-intents/migration/charges#read", - renamed: "addCardViewController(_:didCreatePaymentMethod:completion:)" - ) - @objc optional func addCardViewController( - _ addCardViewController: STPAddCardViewController, - didCreateSource source: STPSource, - completion: STPErrorBlock - ) -} - -private let STPPaymentCardCellReuseIdentifier = "STPPaymentCardCellReuseIdentifier" -enum STPPaymentCardSection: Int { - case stpPaymentCardNumberSection = 0 - case stpPaymentCardScannerSection = 1 - case stpPaymentCardBillingAddressSection = 2 -} - -/// :nodoc: -@_spi(STP) extension STPAddCardViewController: STPAnalyticsProtocol { - @_spi(STP) public static let stp_analyticsIdentifier = "STPAddCardViewController" -} diff --git a/Stripe/StripeiOS/Source/STPAnalyticsClient+BasicUI.swift b/Stripe/StripeiOS/Source/STPAnalyticsClient+BasicUI.swift deleted file mode 100644 index 809037ab7a5..00000000000 --- a/Stripe/StripeiOS/Source/STPAnalyticsClient+BasicUI.swift +++ /dev/null @@ -1,160 +0,0 @@ -// -// STPAnalyticsClient+BasicUI.swift -// StripeiOS -// -// Created by Yuki Tokuhiro on 1/24/24. -// - -import Foundation -@_spi(STP) import StripeCore -@_spi(STP) import StripePayments - -extension STPPaymentContext { - final class AnalyticsLogger { - let analyticsClient = STPAnalyticsClient.sharedClient - var apiClient: STPAPIClient = .shared - var product: String - lazy var commonParameters: [String: Any] = { - [ - "product": product, - ] - }() - - init(product: T.Type) { - self.product = product.stp_analyticsIdentifier - } - - // MARK: - Loading - - func logLoadStarted() { - log(event: .biLoadStarted, params: [:]) - } - - func logLoadSucceeded(loadStartDate: Date, defaultPaymentOption: STPPaymentOption?) { - let event: STPAnalyticEvent = .biLoadSucceeded - let duration = Date().timeIntervalSince(loadStartDate) - let defaultPaymentMethod: String = { - guard let defaultPaymentOption else { - return "none" - } - switch defaultPaymentOption { - case is STPApplePayPaymentOption: - return "apple_pay" - case let defaultPaymentMethod as STPPaymentMethod: - return defaultPaymentMethod.type.identifier - default: - assertionFailure() - return "unknown" - } - }() - let params: [String: Any] = [ - "duration": duration, - "selected_lpm": defaultPaymentMethod, - ] - log(event: event, params: params) - } - - func logLoadFailed(loadStartDate: Date, error: Error) { - let event: STPAnalyticEvent = .biLoadFailed - let duration = Date().timeIntervalSince(loadStartDate) - var params: [String: Any] = [ - "duration": duration, - ] - params.mergeAssertingOnOverwrites(error.serializeForV1Analytics()) - log(event: event, params: params) - } - - // MARK: - Payment - - func logPayment(status: STPPaymentStatus, loadStartDate: Date?, paymentOption: STPPaymentOption, error: Error?) { - let didSucceed: Bool - switch status { - case .userCancellation: - // Don't send analytic for cancels - return - case .success: - didSucceed = true - case .error: - didSucceed = false - @unknown default: - return - } - - let event: STPAnalyticEvent - let paymentMethodType: String - switch paymentOption { - case let paymentMethod as STPPaymentMethod: - paymentMethodType = paymentMethod.type.identifier - event = didSucceed ? .biPaymentCompleteSavedPMSuccess : .biPaymentCompleteSavedPMFailure - case let params as STPPaymentMethodParams: - paymentMethodType = params.type.identifier - event = didSucceed ? .biPaymentCompleteNewPMSuccess : .biPaymentCompleteNewPMFailure - case is STPApplePayPaymentOption: - paymentMethodType = "apple_pay" - event = didSucceed ? .biPaymentCompleteApplePaySuccess : .biPaymentCompleteApplePayFailure - default: - assertionFailure("Unknown payment option!") - return - } - - var params: [String: Any] = ["selected_lpm": paymentMethodType] - if let error { - params.mergeAssertingOnOverwrites(error.serializeForV1Analytics()) - } - if let loadStartDate { - params["duration"] = Date().timeIntervalSince(loadStartDate) - } - - log(event: event, params: params) - } - - // MARK: - UI - - func logPaymentOptionsScreenAppeared() { - log(event: .biOptionsShown, params: [:]) - } - - func logFormShown(paymentMethodType: STPPaymentMethodType) { - let event = STPAnalyticEvent.biFormShown - let params = ["selected_lpm": paymentMethodType] - log(event: event, params: params) - } - - /// - Parameter shownStartDate: The date when the form was first shown. This should never be nil. - func logDoneButtonTapped(paymentMethodType: STPPaymentMethodType, shownStartDate: Date?) { - let event = STPAnalyticEvent.biDoneButtonTapped - - var params: [String: Any] = [ - "selected_lpm": paymentMethodType, - ] - if let shownStartDate { - let duration = Date().timeIntervalSince(shownStartDate) - params["duration"] = duration - } else if NSClassFromString("XCTest") == nil { - assertionFailure("Shown start date should never be nil!") - } - - log(event: event, params: params) - } - - func logFormInteracted(paymentMethodType: STPPaymentMethodType) { - log(event: .biFormInteracted, params: [ - "selected_lpm": paymentMethodType, - ]) - } - - func logCardNumberCompleted() { - log(event: .biCardNumberCompleted, params: [:]) - } - - // MARK: - Helpers - - private func log(event: STPAnalyticEvent, params: [String: Any]) { - let analytic = GenericAnalytic(event: event, params: params.merging(commonParameters, uniquingKeysWith: { new, _ in - assertionFailure("Constructing analytics parameters with duplicate keys") - return new - })) - analyticsClient.log(analytic: analytic, apiClient: apiClient) - } - } -} diff --git a/Stripe/StripeiOS/Source/STPCoreViewController.swift b/Stripe/StripeiOS/Source/STPCoreViewController.swift index 105b544d8e6..20f66879e6b 100644 --- a/Stripe/StripeiOS/Source/STPCoreViewController.swift +++ b/Stripe/StripeiOS/Source/STPCoreViewController.swift @@ -69,23 +69,23 @@ public class STPCoreViewController: UIViewController { _theme = .defaultTheme } - if !useSystemBackButton() { - cancelItem = UIBarButtonItem( - barButtonSystemItem: .cancel, - target: self, - action: #selector(STPAddCardViewController.handleCancelTapped(_:)) - ) - cancelItem?.accessibilityIdentifier = "CoreViewControllerCancelIdentifier" - - stp_navigationItemProxy?.leftBarButtonItem = cancelItem - } - - NotificationCenter.default.addObserver( - self, - selector: #selector(STPAddCardViewController.updateAppearance), - name: UIContentSizeCategory.didChangeNotification, - object: nil - ) +// if !useSystemBackButton() { +// cancelItem = UIBarButtonItem( +// barButtonSystemItem: .cancel, +// target: self, +// action: #selector(STPAddCardViewController.handleCancelTapped(_:)) +// ) +// cancelItem?.accessibilityIdentifier = "CoreViewControllerCancelIdentifier" +// +// stp_navigationItemProxy?.leftBarButtonItem = cancelItem +// } +// +// NotificationCenter.default.addObserver( +// self, +// selector: #selector(STPAddCardViewController.updateAppearance), +// name: UIContentSizeCategory.didChangeNotification, +// object: nil +// ) } /// Called in viewDidLoad after doing base implementation, before diff --git a/Stripe/StripeiOS/Source/STPCustomerContext.swift b/Stripe/StripeiOS/Source/STPCustomerContext.swift deleted file mode 100644 index 13820fa2094..00000000000 --- a/Stripe/StripeiOS/Source/STPCustomerContext.swift +++ /dev/null @@ -1,419 +0,0 @@ -// -// STPCustomerContext.swift -// StripeiOS -// -// Created by Ben Guo on 5/2/17. -// Copyright © 2017 Stripe, Inc. All rights reserved. -// - -import Foundation -@_spi(STP) import StripeCore -@_spi(STP) import StripePaymentsUI - -/// An `STPCustomerContext` retrieves and updates a Stripe customer and their attached -/// payment methods using an ephemeral key, a short-lived API key scoped to a specific -/// customer object. If your current user logs out of your app and a new user logs in, -/// be sure to either create a new instance of `STPCustomerContext` or clear the current -/// instance's cache. On your backend, be sure to create and return a -/// new ephemeral key for the Customer object associated with the new user. -open class STPCustomerContext: NSObject, STPBackendAPIAdapter { - /// Initializes a new `STPCustomerContext` with the specified key provider. - /// Upon initialization, a CustomerContext will fetch a new ephemeral key from - /// your backend and use it to prefetch the customer object specified in the key. - /// Subsequent customer and payment method retrievals (e.g. by `STPPaymentContext`) - /// will return the prefetched customer / attached payment methods immediately if - /// its age does not exceed 60 seconds. - /// - Parameter keyProvider: The key provider the customer context will use. - /// - Returns: the newly-instantiated customer context. - @objc(initWithKeyProvider:) - public convenience init( - keyProvider: STPCustomerEphemeralKeyProvider - ) { - self.init(keyProvider: keyProvider, apiClient: STPAPIClient.shared) - } - - /// Initializes a new `STPCustomerContext` with the specified key provider. - /// Upon initialization, a CustomerContext will fetch a new ephemeral key from - /// your backend and use it to prefetch the customer object specified in the key. - /// Subsequent customer and payment method retrievals (e.g. by `STPPaymentContext`) - /// will return the prefetched customer / attached payment methods immediately if - /// its age does not exceed 60 seconds. - /// - Parameters: - /// - keyProvider: The key provider the customer context will use. - /// - apiClient: The API Client to use to make requests. - /// - Returns: the newly-instantiated customer context. - @objc(initWithKeyProvider:apiClient:) - public convenience init( - keyProvider: STPCustomerEphemeralKeyProvider?, - apiClient: STPAPIClient - ) { - let keyManager = STPEphemeralKeyManager( - keyProvider: keyProvider, - apiVersion: STPAPIClient.apiVersion, - performsEagerFetching: true - ) - self.init(keyManager: keyManager, apiClient: apiClient) - } - - /// `STPCustomerContext` will cache its customer object and associated payment methods - /// for up to 60 seconds. If your current user logs out of your app and a new user logs - /// in, be sure to either call this method or create a new instance of `STPCustomerContext`. - /// On your backend, be sure to create and return a new ephemeral key for the - /// customer object associated with the new user. - @objc - public func clearCache() { - clearCachedCustomer() - clearCachedPaymentMethods() - } - - private var _includeApplePayPaymentMethods = false - /// By default, `STPCustomerContext` will filter Apple Pay when it retrieves - /// Payment Methods. Apple Pay payment methods should generally not be re-used and - /// shouldn't be offered to customers as a new payment method (Apple Pay payment - /// methods may only be re-used for subscriptions). - /// If you are using `STPCustomerContext` to back your own UI and would like to - /// disable Apple Pay filtering, set this property to YES. - /// Note: If you are using `STPPaymentContext`, you should not change this property. - @objc public var includeApplePayPaymentMethods: Bool { - get { - _includeApplePayPaymentMethods - } - set(includeApplePayMethods) { - _includeApplePayPaymentMethods = includeApplePayMethods - customer?.updateSources(filteringApplePay: !includeApplePayMethods) - } - } - - private var _customer: STPCustomer? - private var customer: STPCustomer? { - get { - _customer - } - set(customer) { - _customer = customer - customerRetrievedDate = (customer) != nil ? Date() : nil - } - } - @objc internal var customerRetrievedDate: Date? - - private var _paymentMethods: [STPPaymentMethod]? - private var paymentMethods: [STPPaymentMethod]? { - get { - if !includeApplePayPaymentMethods { - var paymentMethodsExcludingApplePay: [STPPaymentMethod]? = [] - for paymentMethod in _paymentMethods ?? [] { - let isApplePay = - paymentMethod.type == .card && paymentMethod.card?.wallet?.type == .applePay - if !isApplePay { - paymentMethodsExcludingApplePay?.append(paymentMethod) - } - } - return paymentMethodsExcludingApplePay ?? [] - } else { - return _paymentMethods ?? [] - } - } - set(paymentMethods) { - _paymentMethods = paymentMethods - paymentMethodsRetrievedDate = paymentMethods != nil ? Date() : nil - } - } - @objc internal var paymentMethodsRetrievedDate: Date? - private var keyManager: STPEphemeralKeyManagerProtocol - private var apiClient: STPAPIClient - - init( - keyManager: STPEphemeralKeyManagerProtocol, - apiClient: STPAPIClient - ) { - STPAnalyticsClient.sharedClient.addClass(toProductUsageIfNecessary: STPCustomerContext.self) - self.keyManager = keyManager - self.apiClient = apiClient - _includeApplePayPaymentMethods = false - super.init() - retrieveCustomer(nil) - listPaymentMethodsForCustomer(completion: nil) - } - - func clearCachedCustomer() { - customer = nil - } - - func clearCachedPaymentMethods() { - paymentMethods = nil - } - - func shouldUseCachedCustomer() -> Bool { - if customer == nil || customerRetrievedDate == nil { - return false - } - let now = Date() - if let customerRetrievedDate = customerRetrievedDate { - return now.timeIntervalSince(customerRetrievedDate) < CachedCustomerMaxAge - } - return false - } - - func shouldUseCachedPaymentMethods() -> Bool { - if paymentMethods == nil || paymentMethodsRetrievedDate == nil { - return false - } - let now = Date() - if let paymentMethodsRetrievedDate = paymentMethodsRetrievedDate { - return now.timeIntervalSince(paymentMethodsRetrievedDate) < CachedCustomerMaxAge - } - return false - } - - // MARK: - STPBackendAPIAdapter - @objc - public func retrieveCustomer(_ completion: STPCustomerCompletionBlock? = nil) { - if shouldUseCachedCustomer() { - if let completion = completion { - stpDispatchToMainThreadIfNecessary({ - completion(self.customer, nil) - }) - } - return - } - keyManager.getOrCreateKey({ ephemeralKey, retrieveKeyError in - guard let ephemeralKey = ephemeralKey, retrieveKeyError == nil else { - if let completion = completion { - stpDispatchToMainThreadIfNecessary({ - completion(nil, retrieveKeyError) - }) - } - return - } - self.apiClient.retrieveCustomer(using: ephemeralKey) { customer, error in - if let customer = customer { - customer.updateSources(filteringApplePay: !self.includeApplePayPaymentMethods) - self.customer = customer - } - if let completion = completion { - stpDispatchToMainThreadIfNecessary({ - completion(self.customer, error) - }) - } - } - }) - } - - @objc - public func updateCustomer( - withShippingAddress shipping: STPAddress, - completion: STPErrorBlock? - ) { - keyManager.getOrCreateKey({ ephemeralKey, retrieveKeyError in - guard let ephemeralKey = ephemeralKey, retrieveKeyError == nil else { - if let completion = completion { - stpDispatchToMainThreadIfNecessary({ - completion(retrieveKeyError) - }) - } - return - } - var params: [String: Any] = [:] - params["shipping"] = STPAddress.shippingInfoForCharge( - with: shipping, - shippingMethod: nil - ) - self.apiClient.updateCustomer( - withParameters: params, - using: ephemeralKey - ) { customer, error in - if let customer = customer { - customer.updateSources(filteringApplePay: !self.includeApplePayPaymentMethods) - self.customer = customer - } - if let completion = completion { - stpDispatchToMainThreadIfNecessary({ - completion(error) - }) - } - } - }) - } - - /// A convenience method for attaching the PaymentMethod to the current Customer - @objc - public func attachPaymentMethodToCustomer( - paymentMethodId: String, - completion: STPErrorBlock? - ) { - keyManager.getOrCreateKey({ ephemeralKey, retrieveKeyError in - guard let ephemeralKey = ephemeralKey, retrieveKeyError == nil else { - if let completion = completion { - stpDispatchToMainThreadIfNecessary({ - completion(retrieveKeyError) - }) - } - return - } - - self.apiClient.attachPaymentMethod( - paymentMethodId, - toCustomerUsing: ephemeralKey - ) { error in - self.clearCachedPaymentMethods() - if let completion = completion { - stpDispatchToMainThreadIfNecessary({ - completion(error) - }) - } - } - }) - } - - @objc - public func attachPaymentMethod( - toCustomer paymentMethod: STPPaymentMethod, - completion: STPErrorBlock? - ) { - attachPaymentMethodToCustomer( - paymentMethodId: paymentMethod.stripeId, - completion: completion - ) - } - - /// A convenience method for detaching the PaymentMethod to the current Customer - @objc - public func detachPaymentMethodFromCustomer( - paymentMethodId: String, - completion: STPErrorBlock? - ) { - keyManager.getOrCreateKey({ ephemeralKey, retrieveKeyError in - guard let ephemeralKey = ephemeralKey, retrieveKeyError == nil else { - if let completion = completion { - stpDispatchToMainThreadIfNecessary({ - completion(retrieveKeyError) - }) - } - return - } - - self.apiClient.detachPaymentMethod( - paymentMethodId, - fromCustomerUsing: ephemeralKey - ) { error in - self.clearCachedPaymentMethods() - if let completion = completion { - stpDispatchToMainThreadIfNecessary({ - completion(error) - }) - } - } - }) - - } - - @objc - public func detachPaymentMethod( - fromCustomer paymentMethod: STPPaymentMethod, - completion: STPErrorBlock? - ) { - detachPaymentMethodFromCustomer( - paymentMethodId: paymentMethod.stripeId, - completion: completion - ) - } - - @objc - public func listPaymentMethodsForCustomer(completion: STPPaymentMethodsCompletionBlock? = nil) { - if shouldUseCachedPaymentMethods() { - if let completion = completion { - stpDispatchToMainThreadIfNecessary({ - completion(self.paymentMethods, nil) - }) - } - return - } - - keyManager.getOrCreateKey({ ephemeralKey, retrieveKeyError in - guard let ephemeralKey = ephemeralKey, retrieveKeyError == nil else { - if let completion = completion { - stpDispatchToMainThreadIfNecessary({ - completion(nil, retrieveKeyError) - }) - } - return - } - - self.apiClient.listPaymentMethodsForCustomer(using: ephemeralKey) { - paymentMethods, - error in - if paymentMethods != nil { - self.paymentMethods = paymentMethods - } - if let completion = completion { - stpDispatchToMainThreadIfNecessary({ - completion(self.paymentMethods, error) - }) - } - } - }) - } - - func saveLastSelectedPaymentMethodID( - forCustomer paymentMethodID: String?, - completion: STPErrorBlock? - ) { - keyManager.getOrCreateKey({ ephemeralKey, retrieveKeyError in - guard let ephemeralKey = ephemeralKey, retrieveKeyError == nil else { - if let completion = completion { - stpDispatchToMainThreadIfNecessary({ - completion(retrieveKeyError) - }) - } - return - } - - var customerToDefaultPaymentMethodID = - (UserDefaults.standard.dictionary(forKey: kLastSelectedPaymentMethodDefaultsKey)) - as? [String: String] ?? [:] - if let customerID = ephemeralKey.customerID { - customerToDefaultPaymentMethodID[customerID] = paymentMethodID - UserDefaults.standard.set( - customerToDefaultPaymentMethodID, - forKey: kLastSelectedPaymentMethodDefaultsKey - ) - } - - if let completion = completion { - stpDispatchToMainThreadIfNecessary({ - completion(nil) - }) - } - }) - } - - func retrieveLastSelectedPaymentMethodIDForCustomer( - completion: @escaping (String?, Error?) -> Void - ) { - keyManager.getOrCreateKey({ ephemeralKey, retrieveKeyError in - guard let ephemeralKey = ephemeralKey, retrieveKeyError == nil else { - stpDispatchToMainThreadIfNecessary({ - completion(nil, retrieveKeyError) - }) - return - } - - let customerToDefaultPaymentMethodID = - (UserDefaults.standard.dictionary(forKey: kLastSelectedPaymentMethodDefaultsKey)) - as? [String: String] ?? [:] - stpDispatchToMainThreadIfNecessary({ - completion(customerToDefaultPaymentMethodID[ephemeralKey.customerID ?? ""], nil) - }) - }) - } -} - -/// Stores the key we use in NSUserDefaults to save a dictionary of Customer id to their last selected payment method ID -private let kLastSelectedPaymentMethodDefaultsKey = - UserDefaults.StripeKeys.customerToLastSelectedPaymentMethod.rawValue -private let CachedCustomerMaxAge: TimeInterval = 60 - -/// :nodoc: -@_spi(STP) extension STPCustomerContext: STPAnalyticsProtocol { - @_spi(STP) public static var stp_analyticsIdentifier = "STPCustomerContext" -} diff --git a/Stripe/StripeiOS/Source/STPPaymentContext.swift b/Stripe/StripeiOS/Source/STPPaymentContext.swift deleted file mode 100644 index 45af07a5155..00000000000 --- a/Stripe/StripeiOS/Source/STPPaymentContext.swift +++ /dev/null @@ -1,1266 +0,0 @@ -// -// STPPaymentContext.swift -// StripeiOS -// -// Created by Jack Flintermann on 4/20/16. -// Copyright © 2016 Stripe, Inc. All rights reserved. -// - -import Foundation -import ObjectiveC -import PassKit -@_spi(STP) import StripeCore -@_spi(STP) import StripePaymentsUI - -/// An `STPPaymentContext` keeps track of all of the state around a payment. It will manage fetching a user's saved payment methods, tracking any information they select, and prompting them for required additional information before completing their purchase. It can be used to power your application's "payment confirmation" page with just a few lines of code. -/// `STPPaymentContext` also provides a unified interface to multiple payment methods - for example, you can write a single integration to accept both credit card payments and Apple Pay. -/// `STPPaymentContext` saves information about a user's payment methods to a Stripe customer object, and requires an `STPCustomerContext` to manage retrieving and modifying the customer. -@objc(STPPaymentContext) -public class STPPaymentContext: NSObject, STPAuthenticationContext, - STPPaymentOptionsViewControllerDelegate, STPShippingAddressViewControllerDelegate -{ - /// This is a convenience initializer; it is equivalent to calling - /// `init(customerContext:customerContext - /// configuration:STPPaymentConfiguration.shared - /// theme:STPTheme.defaultTheme`. - /// - Parameter customerContext: The customer context the payment context will use to fetch - /// and modify its Stripe customer. - seealso: STPCustomerContext.h - /// - Returns: the newly-instantiated payment context - @objc - public convenience init( - customerContext: STPCustomerContext - ) { - self.init(apiAdapter: customerContext) - } - - /// Initializes a new Payment Context with the provided customer context, configuration, - /// and theme. After this class is initialized, you should also make sure to set its - /// `delegate` and `hostViewController` properties. - /// - Parameters: - /// - customerContext: The customer context the payment context will use to fetch - /// and modify its Stripe customer. - seealso: STPCustomerContext.h - /// - configuration: The configuration for the payment context to use. This - /// lets you set your Stripe publishable API key, required billing address fields, etc. - /// - seealso: STPPaymentConfiguration.h - /// - theme: The theme describing the visual appearance of all UI - /// that the payment context automatically creates for you. - seealso: STPTheme.h - /// - Returns: the newly-instantiated payment context - @objc - public convenience init( - customerContext: STPCustomerContext, - configuration: STPPaymentConfiguration, - theme: STPTheme - ) { - self.init( - apiAdapter: customerContext, - configuration: configuration, - theme: theme - ) - } - - /// Note: Instead of providing your own backend API adapter, we recommend using - /// `STPCustomerContext`, which will manage retrieving and updating a - /// Stripe customer for you. - seealso: STPCustomerContext.h - /// This is a convenience initializer; it is equivalent to calling - /// `init(apiAdapter:apiAdapter configuration:STPPaymentConfiguration.shared theme:STPTheme.defaultTheme)`. - @objc - public convenience init( - apiAdapter: STPBackendAPIAdapter - ) { - self.init( - apiAdapter: apiAdapter, - configuration: STPPaymentConfiguration.shared, - theme: STPTheme.defaultTheme - ) - } - - /// Note: Instead of providing your own backend API adapter, we recommend using - /// `STPCustomerContext`, which will manage retrieving and updating a - /// Stripe customer for you. - seealso: STPCustomerContext.h - /// Initializes a new Payment Context with the provided API adapter and configuration. - /// After this class is initialized, you should also make sure to set its `delegate` - /// and `hostViewController` properties. - /// - Parameters: - /// - apiAdapter: The API adapter the payment context will use to fetch and - /// modify its contents. You need to make a class conforming to this protocol that - /// talks to your server. - seealso: STPBackendAPIAdapter.h - /// - configuration: The configuration for the payment context to use. This lets - /// you set your Stripe publishable API key, required billing address fields, etc. - /// - seealso: STPPaymentConfiguration.h - /// - theme: The theme describing the visual appearance of all UI that - /// the payment context automatically creates for you. - seealso: STPTheme.h - /// - Returns: the newly-instantiated payment context - @objc - public init( - apiAdapter: STPBackendAPIAdapter, - configuration: STPPaymentConfiguration, - theme: STPTheme - ) { - STPAnalyticsClient.sharedClient.addClass(toProductUsageIfNecessary: STPPaymentContext.self) - AnalyticsHelper.shared.generateSessionID() - self.configuration = configuration - self.apiAdapter = apiAdapter - self.theme = theme - paymentCurrency = "USD" - paymentCountry = "US" - apiClient = STPAPIClient.shared - modalPresentationStyle = .fullScreen - state = STPPaymentContextState.none - super.init() - retryLoading() - } - - /// Note: Instead of providing your own backend API adapter, we recommend using - /// `STPCustomerContext`, which will manage retrieving and updating a - /// Stripe customer for you. - seealso: STPCustomerContext.h - /// The API adapter the payment context will use to fetch and modify its contents. - /// You need to make a class conforming to this protocol that talks to your server. - /// - seealso: STPBackendAPIAdapter.h - @objc public private(set) var apiAdapter: STPBackendAPIAdapter - /// The configuration for the payment context to use internally. - seealso: STPPaymentConfiguration.h - @objc public private(set) var configuration: STPPaymentConfiguration - /// The visual appearance that will be used by any views that the context generates. - seealso: STPTheme.h - @objc public private(set) var theme: STPTheme - - private var _prefilledInformation: STPUserInformation? - /// If you've already collected some information from your user, you can set it here and it'll be automatically filled out when possible/appropriate in any UI that the payment context creates. - @objc public var prefilledInformation: STPUserInformation? { - get { - _prefilledInformation - } - set(prefilledInformation) { - _prefilledInformation = prefilledInformation - if prefilledInformation?.shippingAddress != nil && shippingAddress == nil { - shippingAddress = prefilledInformation?.shippingAddress - shippingAddressNeedsVerification = true - } - } - } - - private weak var _hostViewController: UIViewController? - /// The view controller that any additional UI will be presented on. If you have a "checkout view controller" in your app, that should be used as the host view controller. - @objc public weak var hostViewController: UIViewController? { - get { - _hostViewController - } - set(hostViewController) { - assert( - _hostViewController == nil, - "You cannot change the hostViewController on an STPPaymentContext after it's already been set." - ) - _hostViewController = hostViewController - if hostViewController is UINavigationController { - originalTopViewController = - (hostViewController as? UINavigationController)?.topViewController - } - if let hostViewController = hostViewController { - artificiallyRetain(hostViewController) - } - } - } - - private weak var _delegate: STPPaymentContextDelegate? - /// This delegate will be notified when the payment context's contents change. - seealso: STPPaymentContextDelegate - @objc public weak var delegate: STPPaymentContextDelegate? { - get { - _delegate - } - set(delegate) { - _delegate = delegate - DispatchQueue.main.async(execute: { - self.delegate?.paymentContextDidChange(self) - }) - } - } - /// Whether or not the payment context is currently loading information from the network. - - @objc public var loading: Bool { - return !(loadingPromise?.completed)! - } - /// @note This is no longer recommended as of v18.3.0 - the SDK automatically saves the Stripe ID of the last selected - /// payment method using NSUserDefaults and displays it as the default pre-selected option. You can override this behavior - /// by setting this property. - /// The Stripe ID of a payment method to display as the default pre-selected option. - /// @note Set this property immediately after initializing STPPaymentContext, or call `retryLoading` afterwards. - @objc public var defaultPaymentMethod: String? - - private var _selectedPaymentOption: STPPaymentOption? - /// The user's currently selected payment option. May be nil. - @objc public private(set) var selectedPaymentOption: STPPaymentOption? { - get { - _selectedPaymentOption - } - set { - if let newValue = newValue, let paymentOptions = self.paymentOptions { - if !paymentOptions.contains(where: { (option) -> Bool in - newValue.isEqual(option) - }) { - if newValue.isReusable { - self.paymentOptions = paymentOptions + [newValue] - } - } - } - if !(_selectedPaymentOption?.isEqual(newValue) ?? false) { - _selectedPaymentOption = newValue - stpDispatchToMainThreadIfNecessary({ - self.delegate?.paymentContextDidChange(self) - }) - } - - } - } - - private var _paymentOptions: [STPPaymentOption]? - /// The available payment options the user can choose between. May be nil. - @objc public private(set) var paymentOptions: [STPPaymentOption]? { - get { - _paymentOptions - } - set { - _paymentOptions = newValue?.sorted(by: { (obj1, obj2) -> Bool in - let applePayKlass = STPApplePayPaymentOption.self - let paymentMethodKlass = STPPaymentMethod.self - if obj1.isKind(of: applePayKlass) { - return true - } else if obj2.isKind(of: applePayKlass) { - return false - } - if obj1.isKind(of: paymentMethodKlass) && obj2.isKind(of: paymentMethodKlass) { - return (obj1.label.compare(obj2.label) == .orderedAscending) - } - return false - }) - } - } - - /// The user's currently selected shipping method. May be nil. - @objc public internal(set) var selectedShippingMethod: PKShippingMethod? - - private var _shippingMethods: [PKShippingMethod]? - /// An array of STPShippingMethod objects that describe the supported shipping methods. May be nil. - @objc public private(set) var shippingMethods: [PKShippingMethod]? { - get { - _shippingMethods - } - set { - _shippingMethods = newValue - if let shippingMethods = newValue, - let selectedShippingMethod = self.selectedShippingMethod - { - if shippingMethods.count == 0 { - self.selectedShippingMethod = nil - } else if shippingMethods.contains(selectedShippingMethod) { - self.selectedShippingMethod = shippingMethods.first - } - } - } - } - - /// The user's shipping address. May be nil. - /// If you've already collected a shipping address from your user, you may - /// prefill it by setting a shippingAddress in PaymentContext's prefilledInformation. - /// When your user enters a new shipping address, PaymentContext will save it to - /// the current customer object. When PaymentContext loads, if you haven't - /// manually set a prefilled value, any shipping information saved on the customer - /// will be used to prefill the shipping address form. Note that because your - /// customer's email may not be the same as the email provided with their shipping - /// info, PaymentContext will not prefill the shipping form's email using your - /// customer's email. - /// You should not rely on the shipping information stored on the Stripe customer - /// for order fulfillment, as your user may change this information if they make - /// multiple purchases. We recommend adding shipping information when you create - /// a charge (which can also help prevent fraud), or saving it to your own - /// database. https://stripe.com/docs/api/payment_intents/create#create_payment_intent-shipping - /// Note: by default, your user will still be prompted to verify a prefilled - /// shipping address. To change this behavior, you can set - /// `verifyPrefilledShippingAddress` to NO in your `STPPaymentConfiguration`. - @objc public private(set) var shippingAddress: STPAddress? - /// The amount of money you're requesting from the user, in the smallest currency - /// unit for the selected currency. For example, to indicate $10 USD, use 1000 - /// (i.e. 1000 cents). For more information, see https://stripe.com/docs/api/payment_intents/create#create_payment_intent-amount - /// @note This value must be present and greater than zero in order for Apple Pay - /// to be automatically enabled. - /// @note You should only set either this or `paymentSummaryItems`, not both. - /// The other will be automatically calculated on demand using your `paymentCurrency`. - - @objc public var paymentAmount: Int { - get { - return paymentAmountModel.paymentAmount( - withCurrency: paymentCurrency, - shippingMethod: selectedShippingMethod - ) - } - set(paymentAmount) { - paymentAmountModel = STPPaymentContextAmountModel(amount: paymentAmount) - } - } - /// The three-letter currency code for the currency of the payment (i.e. USD, GBP, - /// JPY, etc). Defaults to "USD". - /// @note Changing this property may change the return value of `paymentAmount` - /// or `paymentSummaryItems` (whichever one you didn't directly set yourself). - @objc public var paymentCurrency: String - /// The two-letter country code for the country where the payment will be processed. - /// You should set this to the country your Stripe account is in. Defaults to "US". - /// @note Changing this property will change the `countryCode` of your Apple Pay - /// payment requests. - /// - seealso: PKPaymentRequest for more information. - @objc public var paymentCountry: String - /// If you support Apple Pay, you can optionally set the PKPaymentSummaryItems - /// you want to display here instead of using `paymentAmount`. Note that the - /// grand total (the amount of the last summary item) must be greater than zero. - /// If not set, a single summary item will be automatically generated using - /// `paymentAmount` and your configuration's `companyName`. - /// - seealso: PKPaymentRequest for more information - /// @note You should only set either this or `paymentAmount`, not both. - /// The other will be automatically calculated on demand using your `paymentCurrency.` - - @objc public var paymentSummaryItems: [PKPaymentSummaryItem] { - get { - return paymentAmountModel.paymentSummaryItems( - withCurrency: paymentCurrency, - companyName: configuration.companyName, - shippingMethod: selectedShippingMethod - ) ?? [] - } - set(paymentSummaryItems) { - paymentAmountModel = STPPaymentContextAmountModel( - paymentSummaryItems: paymentSummaryItems - ) - } - } - - /// A value that indicates whether Apple Pay Later is available for a transaction. - /// Defaults to enabled. - /// - Seealso: This property is mirrors `PKPaymentRequest.applePayLaterAvailability` -#if compiler(>=5.9) - @available(macOS 14.0, iOS 17.0, *) - @objc public var applePayLaterAvailability: PKApplePayLaterAvailability { - // Stored properties cannot be marked potentially unavailable with '@available', so do this workaround instead - get { - return _applePayLaterAvailability as! PKApplePayLaterAvailability - } - set { - _applePayLaterAvailability = newValue - - } - } - private lazy var _applePayLaterAvailability: Any? = { - if #available(macOS 14.0, iOS 17.0, *) { - return PKApplePayLaterAvailability.available - } - return nil - }() -#endif - /// The presentation style used for all view controllers presented modally by the context. - /// Since custom transition styles are not supported, you should set this to either - /// `UIModalPresentationFullScreen`, `UIModalPresentationPageSheet`, or `UIModalPresentationFormSheet`. - /// The default value is `UIModalPresentationFullScreen`. - @objc public var modalPresentationStyle: UIModalPresentationStyle = .fullScreen - /// The mode to use when displaying the title of the navigation bar in all view - /// controllers presented by the context. The default value is `automatic`, - /// which causes the title to use the same styling as the previously displayed - /// navigation item (if the view controller is pushed onto the `hostViewController`). - /// If the `prefersLargeTitles` property of the `hostViewController`'s navigation bar - /// is false, this property has no effect and the navigation item's title is always - /// displayed as a small title. - /// If the view controller is presented modally, `automatic` and - /// `never` always result in a navigation bar with a small title. - @objc public var largeTitleDisplayMode = UINavigationItem.LargeTitleDisplayMode.automatic - /// A view that will be placed as the footer of the payment options selection - /// view controller. - /// When the footer view needs to be resized, it will be sent a - /// `sizeThatFits:` call. The view should respond correctly to this method in order - /// to be sized and positioned properly. - @objc public var paymentOptionsViewControllerFooterView: UIView? - /// A view that will be placed as the footer of the add card view controller. - /// When the footer view needs to be resized, it will be sent a - /// `sizeThatFits:` call. The view should respond correctly to this method in order - /// to be sized and positioned properly. - @objc public var addCardViewControllerFooterView: UIView? - /// The API Client to use to make requests. - /// Defaults to STPAPIClient.shared - public var apiClient: STPAPIClient = .shared { - didSet { - analyticsLogger.apiClient = apiClient - } - } - internal let analyticsLogger: AnalyticsLogger = .init(product: STPPaymentContext.self) - internal var loadingStartDate: Date? - - /// If `paymentContext:didFailToLoadWithError:` is called on your delegate, you - /// can in turn call this method to try loading again (if that hasn't been called, - /// calling this will do nothing). If retrying in turn fails, `paymentContext:didFailToLoadWithError:` - /// will be called again (and you can again call this to keep retrying, etc). - @objc - public func retryLoading() { - let loadingStartDate = Date() - self.loadingStartDate = loadingStartDate - analyticsLogger.logLoadStarted() - // Clear any cached customer object and attached payment methods before refetching - if apiAdapter is STPCustomerContext { - let customerContext = apiAdapter as? STPCustomerContext - customerContext?.clearCache() - } - weak var weakSelf = self - loadingPromise = STPPromise.init().onSuccess({ tuple in - guard let strongSelf = weakSelf else { - return - } - strongSelf.analyticsLogger.logLoadSucceeded(loadStartDate: loadingStartDate, defaultPaymentOption: tuple.selectedPaymentOption) - strongSelf.paymentOptions = tuple.paymentOptions - strongSelf.selectedPaymentOption = tuple.selectedPaymentOption - }).onFailure({ error in - guard let strongSelf = weakSelf else { - return - } - strongSelf.analyticsLogger.logLoadFailed(loadStartDate: loadingStartDate, error: error) - if strongSelf.hostViewController != nil { - if strongSelf.paymentOptionsViewController != nil - && strongSelf.paymentOptionsViewController?.viewIfLoaded?.window != nil - { - if let paymentOptionsViewController1 = strongSelf.paymentOptionsViewController { - strongSelf.appropriatelyDismiss(paymentOptionsViewController1) { - strongSelf.delegate?.paymentContext( - strongSelf, - didFailToLoadWithError: error - ) - } - } - } else { - strongSelf.delegate?.paymentContext(strongSelf, didFailToLoadWithError: error) - } - } - }) - apiAdapter.retrieveCustomer({ customer, retrieveCustomerError in - stpDispatchToMainThreadIfNecessary({ - guard let strongSelf = weakSelf else { - return - } - if let retrieveCustomerError = retrieveCustomerError { - strongSelf.loadingPromise?.fail(retrieveCustomerError) - return - } - if strongSelf.shippingAddress == nil && customer?.shippingAddress != nil { - strongSelf.shippingAddress = customer?.shippingAddress - strongSelf.shippingAddressNeedsVerification = true - } - - strongSelf.apiAdapter.listPaymentMethodsForCustomer(completion: { - paymentMethods, - error in - guard let strongSelf2 = weakSelf else { - return - } - stpDispatchToMainThreadIfNecessary({ - if let error = error { - strongSelf2.loadingPromise?.fail(error) - return - } - - if self.defaultPaymentMethod == nil - && (strongSelf2.apiAdapter is STPCustomerContext) - { - // Retrieve the last selected payment method saved by STPCustomerContext - (strongSelf2.apiAdapter as? STPCustomerContext)? - .retrieveLastSelectedPaymentMethodIDForCustomer(completion: { - paymentMethodID, - _ in - guard let strongSelf3 = weakSelf else { - return - } - if let paymentMethods = paymentMethods { - let paymentTuple = STPPaymentOptionTuple( - filteredForUIWith: paymentMethods, - selectedPaymentMethod: paymentMethodID, - configuration: strongSelf3.configuration - ) - strongSelf3.loadingPromise?.succeed(paymentTuple) - } else { - strongSelf3.loadingPromise?.fail( - STPErrorCode.invalidRequestError as! Error - ) - } - }) - } else { - if let paymentMethods = paymentMethods { - let paymentTuple = STPPaymentOptionTuple( - filteredForUIWith: paymentMethods, - selectedPaymentMethod: self.defaultPaymentMethod, - configuration: strongSelf2.configuration - ) - strongSelf2.loadingPromise?.succeed(paymentTuple) - } - } - }) - }) - }) - }) - } - - /// This creates, configures, and appropriately presents an `STPPaymentOptionsViewController` - /// on top of the payment context's `hostViewController`. It'll be dismissed automatically - /// when the user is done selecting their payment method. - /// @note This method will do nothing if it is called while STPPaymentContext is - /// already showing a view controller or in the middle of requesting a payment. - @objc - public func presentPaymentOptionsViewController() { - presentPaymentOptionsViewController(withNewState: .showingRequestedViewController) - } - - /// This creates, configures, and appropriately pushes an `STPPaymentOptionsViewController` - /// onto the navigation stack of the context's `hostViewController`. It'll be popped - /// automatically when the user is done selecting their payment method. - /// @note This method will do nothing if it is called while STPPaymentContext is - /// already showing a view controller or in the middle of requesting a payment. - @objc - public func pushPaymentOptionsViewController() { - assert( - hostViewController != nil && hostViewController?.viewIfLoaded?.window != nil, - "hostViewController must not be nil on STPPaymentContext when calling pushPaymentOptionsViewController on it. Next time, set the hostViewController property first!" - ) - var navigationController: UINavigationController? - if hostViewController is UINavigationController { - navigationController = hostViewController as? UINavigationController - } else { - navigationController = hostViewController?.navigationController - } - assert( - navigationController != nil, - "The payment context's hostViewController is not a navigation controller, or is not contained in one. Either make sure it is inside a navigation controller before calling pushPaymentOptionsViewController, or call presentPaymentOptionsViewController instead." - ) - if state == STPPaymentContextState.none { - state = .showingRequestedViewController - - let paymentOptionsViewController = STPPaymentOptionsViewController(paymentContext: self) - self.paymentOptionsViewController = paymentOptionsViewController - paymentOptionsViewController.prefilledInformation = prefilledInformation - paymentOptionsViewController.defaultPaymentMethod = defaultPaymentMethod - paymentOptionsViewController.paymentOptionsViewControllerFooterView = - paymentOptionsViewControllerFooterView - paymentOptionsViewController.addCardViewControllerFooterView = - addCardViewControllerFooterView - paymentOptionsViewController.navigationItem.largeTitleDisplayMode = - largeTitleDisplayMode - - navigationController?.pushViewController( - paymentOptionsViewController, - animated: transitionAnimationsEnabled() - ) - } - } - - /// This creates, configures, and appropriately presents a view controller for - /// collecting shipping address and shipping method on top of the payment context's - /// `hostViewController`. It'll be dismissed automatically when the user is done - /// entering their shipping info. - /// @note This method will do nothing if it is called while STPPaymentContext is - /// already showing a view controller or in the middle of requesting a payment. - @objc - public func presentShippingViewController() { - presentShippingViewController(withNewState: .showingRequestedViewController) - } - - /// This creates, configures, and appropriately pushes a view controller for - /// collecting shipping address and shipping method onto the navigation stack of - /// the context's `hostViewController`. It'll be popped automatically when the - /// user is done entering their shipping info. - /// @note This method will do nothing if it is called while STPPaymentContext is - /// already showing a view controller, or in the middle of requesting a payment. - @objc - public func pushShippingViewController() { - assert( - hostViewController != nil && hostViewController?.viewIfLoaded?.window != nil, - "hostViewController must not be nil on STPPaymentContext when calling pushShippingViewController on it. Next time, set the hostViewController property first!" - ) - var navigationController: UINavigationController? - if hostViewController is UINavigationController { - navigationController = hostViewController as? UINavigationController - } else { - navigationController = hostViewController?.navigationController - } - assert( - navigationController != nil, - "The payment context's hostViewController is not a navigation controller, or is not contained in one. Either make sure it is inside a navigation controller before calling pushShippingInfoViewController, or call presentShippingInfoViewController instead." - ) - if state == STPPaymentContextState.none { - state = .showingRequestedViewController - - let addressViewController = STPShippingAddressViewController(paymentContext: self) - addressViewController.navigationItem.largeTitleDisplayMode = largeTitleDisplayMode - navigationController?.pushViewController( - addressViewController, - animated: transitionAnimationsEnabled() - ) - } - } - - /// Requests payment from the user. This may need to present some supplemental UI - /// to the user, in which case it will be presented on the payment context's - /// `hostViewController`. For instance, if they've selected Apple Pay as their - /// payment method, calling this method will show the payment sheet. If the user - /// has a card on file, this will use that without presenting any additional UI. - /// After this is called, the `paymentContext:didCreatePaymentResult:completion:` - /// and `paymentContext:didFinishWithStatus:error:` methods will be called on the - /// context's `delegate`. - /// @note This method will do nothing if it is called while STPPaymentContext is - /// already showing a view controller, or in the middle of requesting a payment. - @objc - public func requestPayment() { - weak var weakSelf = self - loadingPromise?.onSuccess({ _ in - guard let strongSelf = weakSelf else { - return - } - - if strongSelf.state != STPPaymentContextState.none { - return - } - - if strongSelf.selectedPaymentOption == nil { - strongSelf.presentPaymentOptionsViewController(withNewState: .requestingPayment) - } else if strongSelf.requestPaymentShouldPresentShippingViewController() { - strongSelf.presentShippingViewController(withNewState: .requestingPayment) - } else if let selectedPaymentOption = strongSelf.selectedPaymentOption, - (selectedPaymentOption is STPPaymentMethod || selectedPaymentOption is STPPaymentMethodParams) - { - strongSelf.state = .requestingPayment - let result = STPPaymentResult(paymentOption: strongSelf.selectedPaymentOption!) - strongSelf.delegate?.paymentContext(self, didCreatePaymentResult: result) { status, error in - // Note `selectedPaymentOption` is always an `STPPaymentMethod` for cards - strongSelf.analyticsLogger.logPayment(status: status, loadStartDate: strongSelf.loadingStartDate, paymentOption: selectedPaymentOption, error: error) - stpDispatchToMainThreadIfNecessary({ - strongSelf.didFinish(with: status, error: error) - }) - } - } else if let selectedPaymentOption = strongSelf.selectedPaymentOption, selectedPaymentOption is STPApplePayPaymentOption { - assert( - strongSelf.hostViewController != nil, - "hostViewController must not be nil on STPPaymentContext. Next time, set the hostViewController property first!" - ) - strongSelf.state = .requestingPayment - let paymentRequest = strongSelf.buildPaymentRequest() - let shippingAddressHandler: STPShippingAddressSelectionBlock = { - shippingAddress, - completion in - // Apple Pay always returns a partial address here, so we won't - // update self.shippingAddress or self.shippingMethods - if strongSelf.delegate?.responds( - to: #selector( - STPPaymentContextDelegate.paymentContext( - _: - didUpdateShippingAddress: - completion: - )) - ) - ?? false - { - strongSelf.delegate?.paymentContext?( - strongSelf, - didUpdateShippingAddress: shippingAddress - ) { status, _, shippingMethods, _ in - completion( - status, - shippingMethods ?? [], - strongSelf.paymentSummaryItems - ) - } - } else { - completion( - .valid, - strongSelf.shippingMethods ?? [], - strongSelf.paymentSummaryItems - ) - } - } - let shippingMethodHandler: STPShippingMethodSelectionBlock = { - shippingMethod, - completion in - strongSelf.selectedShippingMethod = shippingMethod - strongSelf.delegate?.paymentContextDidChange(strongSelf) - completion(self.paymentSummaryItems) - } - let paymentHandler: STPPaymentAuthorizationBlock = { payment in - strongSelf.selectedShippingMethod = payment.shippingMethod - if let shippingContact = payment.shippingContact { - strongSelf.shippingAddress = STPAddress(pkContact: shippingContact) - } - strongSelf.shippingAddressNeedsVerification = false - strongSelf.delegate?.paymentContextDidChange(strongSelf) - if strongSelf.apiAdapter is STPCustomerContext { - let customerContext = strongSelf.apiAdapter as? STPCustomerContext - if let shippingAddress1 = strongSelf.shippingAddress { - customerContext?.updateCustomer( - withShippingAddress: shippingAddress1, - completion: nil - ) - } - } - } - let applePayPaymentMethodHandler: STPApplePayPaymentMethodHandlerBlock = { - paymentMethod, - completion in - strongSelf.apiAdapter.attachPaymentMethod(toCustomer: paymentMethod) { - attachPaymentMethodError in - stpDispatchToMainThreadIfNecessary({ - if attachPaymentMethodError != nil { - completion(.error, attachPaymentMethodError) - } else { - let result = STPPaymentResult(paymentOption: paymentMethod) - strongSelf.delegate?.paymentContext( - strongSelf, - didCreatePaymentResult: result - ) { - status, - error in - // for Apple Pay, the didFinishWithStatus callback is fired later when Apple Pay VC finishes - completion(status, error) - } - } - }) - } - } - if let paymentRequest = paymentRequest { - strongSelf.applePayVC = PKPaymentAuthorizationViewController.stp_controller( - with: paymentRequest, - apiClient: strongSelf.apiClient, - onShippingAddressSelection: shippingAddressHandler, - onShippingMethodSelection: shippingMethodHandler, - onPaymentAuthorization: paymentHandler, - onPaymentMethodCreation: applePayPaymentMethodHandler, - onFinish: { status, error in - strongSelf.analyticsLogger.logPayment(status: status, loadStartDate: strongSelf.loadingStartDate, paymentOption: selectedPaymentOption, error: error) - if strongSelf.applePayVC?.presentingViewController != nil { - strongSelf.hostViewController?.dismiss( - animated: strongSelf.transitionAnimationsEnabled() - ) { - strongSelf.didFinish(with: status, error: error) - } - } else { - strongSelf.didFinish(with: status, error: error) - } - strongSelf.applePayVC = nil - } - ) - } - if let applePayVC1 = strongSelf.applePayVC { - strongSelf.hostViewController?.present( - applePayVC1, - animated: strongSelf.transitionAnimationsEnabled() - ) - } - } - }).onFailure({ error in - guard let strongSelf = weakSelf else { - return - } - strongSelf.didFinish(with: .error, error: error) - }) - } - private var loadingPromise: STPPromise? - private weak var paymentOptionsViewController: STPPaymentOptionsViewController? - private var state: STPPaymentContextState = .none - private var paymentAmountModel = STPPaymentContextAmountModel(amount: 0) - private var shippingAddressNeedsVerification = false - // If hostViewController was set to a nav controller, the original VC on top of the stack - private weak var originalTopViewController: UIViewController? - private var applePayVC: PKPaymentAuthorizationViewController? - - // Disable transition animations in tests - func transitionAnimationsEnabled() -> Bool { - return NSClassFromString("XCTest") == nil - } - - var currentValuePromise: STPPromise { - weak var weakSelf = self - return - (loadingPromise?.map({ _ in - guard let strongSelf = weakSelf, let paymentOptions = strongSelf.paymentOptions - else { - return STPPaymentOptionTuple() - } - return STPPaymentOptionTuple( - paymentOptions: paymentOptions, - selectedPaymentOption: strongSelf.selectedPaymentOption - ) - }))! - } - - func remove(_ paymentOptionToRemove: STPPaymentOption?) { - // Remove payment method from cached representation - var paymentOptions = self.paymentOptions - paymentOptions?.removeAll { $0 as AnyObject === paymentOptionToRemove as AnyObject } - self.paymentOptions = paymentOptions - - // Elect new selected payment method if needed - if let selectedPaymentOption = selectedPaymentOption, - selectedPaymentOption.isEqual(paymentOptionToRemove) - { - self.selectedPaymentOption = self.paymentOptions?.first - } - } - - // MARK: - Payment Methods - - func presentPaymentOptionsViewController(withNewState state: STPPaymentContextState) { - assert( - hostViewController != nil && hostViewController?.viewIfLoaded?.window != nil, - "hostViewController must not be nil on STPPaymentContext when calling pushPaymentOptionsViewController on it. Next time, set the hostViewController property first!" - ) - if self.state == STPPaymentContextState.none { - self.state = state - let paymentOptionsViewController = STPPaymentOptionsViewController(paymentContext: self) - self.paymentOptionsViewController = paymentOptionsViewController - paymentOptionsViewController.prefilledInformation = prefilledInformation - paymentOptionsViewController.defaultPaymentMethod = defaultPaymentMethod - paymentOptionsViewController.paymentOptionsViewControllerFooterView = - paymentOptionsViewControllerFooterView - paymentOptionsViewController.addCardViewControllerFooterView = - addCardViewControllerFooterView - paymentOptionsViewController.navigationItem.largeTitleDisplayMode = - largeTitleDisplayMode - - let navigationController = UINavigationController( - rootViewController: paymentOptionsViewController - ) - navigationController.navigationBar.stp_theme = theme - navigationController.navigationBar.prefersLargeTitles = true - navigationController.modalPresentationStyle = modalPresentationStyle - hostViewController?.present( - navigationController, - animated: transitionAnimationsEnabled() - ) - } - } - - @objc - public func paymentOptionsViewController( - _ paymentOptionsViewController: STPPaymentOptionsViewController, - didSelect paymentOption: STPPaymentOption - ) { - selectedPaymentOption = paymentOption - } - - @objc - public func paymentOptionsViewControllerDidFinish( - _ paymentOptionsViewController: STPPaymentOptionsViewController - ) { - appropriatelyDismiss(paymentOptionsViewController) { - if self.state == .requestingPayment { - self.state = STPPaymentContextState.none - self.requestPayment() - } else { - self.state = STPPaymentContextState.none - } - } - } - - @objc - public func paymentOptionsViewControllerDidCancel( - _ paymentOptionsViewController: STPPaymentOptionsViewController - ) { - appropriatelyDismiss(paymentOptionsViewController) { - if self.state == .requestingPayment { - self.didFinish( - with: .userCancellation, - error: nil - ) - } else { - self.state = STPPaymentContextState.none - } - } - } - - @objc - public func paymentOptionsViewController( - _ paymentOptionsViewController: STPPaymentOptionsViewController, - didFailToLoadWithError error: Error - ) { - // we'll handle this ourselves when the loading promise fails. - } - - @objc(appropriatelyDismissPaymentOptionsViewController:completion:) func appropriatelyDismiss( - _ viewController: STPPaymentOptionsViewController, - completion: @escaping STPVoidBlock - ) { - if viewController.stp_isAtRootOfNavigationController() { - // if we're the root of the navigation controller, we've been presented modally. - viewController.presentingViewController?.dismiss( - animated: transitionAnimationsEnabled() - ) { - self.paymentOptionsViewController = nil - completion() - } - } else { - // otherwise, we've been pushed onto the stack. - var destinationViewController = hostViewController - // If hostViewController is a nav controller, pop to the original VC on top of the stack. - if hostViewController is UINavigationController { - destinationViewController = originalTopViewController - } - viewController.navigationController?.stp_pop( - to: destinationViewController, - animated: transitionAnimationsEnabled() - ) { - self.paymentOptionsViewController = nil - completion() - } - } - } - - // MARK: - Shipping Info - - func presentShippingViewController(withNewState state: STPPaymentContextState) { - assert( - hostViewController != nil && hostViewController?.viewIfLoaded?.window != nil, - "hostViewController must not be nil on STPPaymentContext when calling presentShippingViewController on it. Next time, set the hostViewController property first!" - ) - - if self.state == STPPaymentContextState.none { - self.state = state - - let addressViewController = STPShippingAddressViewController(paymentContext: self) - addressViewController.navigationItem.largeTitleDisplayMode = largeTitleDisplayMode - let navigationController = UINavigationController( - rootViewController: addressViewController - ) - navigationController.navigationBar.stp_theme = theme - navigationController.navigationBar.prefersLargeTitles = true - navigationController.modalPresentationStyle = modalPresentationStyle - hostViewController?.present( - navigationController, - animated: transitionAnimationsEnabled() - ) - } - } - - @objc - public func shippingAddressViewControllerDidCancel( - _ addressViewController: STPShippingAddressViewController - ) { - appropriatelyDismiss(addressViewController) { - if self.state == .requestingPayment { - self.didFinish( - with: .userCancellation, - error: nil - ) - } else { - self.state = STPPaymentContextState.none - } - } - } - - @objc - public func shippingAddressViewController( - _ addressViewController: STPShippingAddressViewController, - didEnter address: STPAddress, - completion: @escaping STPShippingMethodsCompletionBlock - ) { - if delegate?.responds( - to: #selector( - STPPaymentContextDelegate.paymentContext(_:didUpdateShippingAddress:completion:)) - ) - ?? false - { - delegate?.paymentContext?(self, didUpdateShippingAddress: address) { - status, - shippingValidationError, - shippingMethods, - selectedMethod in - self.shippingMethods = shippingMethods - completion(status, shippingValidationError, shippingMethods, selectedMethod) - } - } else { - completion(.valid, nil, nil, nil) - } - } - - @objc - public func shippingAddressViewController( - _ addressViewController: STPShippingAddressViewController, - didFinishWith address: STPAddress, - shippingMethod method: PKShippingMethod? - ) { - shippingAddress = address - shippingAddressNeedsVerification = false - selectedShippingMethod = method - delegate?.paymentContextDidChange(self) - if apiAdapter.responds( - to: #selector(STPCustomerContext.updateCustomer(withShippingAddress:completion:)) - ) { - if let shippingAddress = shippingAddress { - apiAdapter.updateCustomer?(withShippingAddress: shippingAddress, completion: nil) - } - } - appropriatelyDismiss(addressViewController) { - if self.state == .requestingPayment { - self.state = STPPaymentContextState.none - self.requestPayment() - } else { - self.state = STPPaymentContextState.none - } - } - } - - @objc(appropriatelyDismissViewController:completion:) func appropriatelyDismiss( - _ viewController: UIViewController, - completion: @escaping STPVoidBlock - ) { - if viewController.stp_isAtRootOfNavigationController() { - // if we're the root of the navigation controller, we've been presented modally. - viewController.presentingViewController?.dismiss( - animated: transitionAnimationsEnabled() - ) { - completion() - } - } else { - // otherwise, we've been pushed onto the stack. - var destinationViewController = hostViewController - // If hostViewController is a nav controller, pop to the original VC on top of the stack. - if hostViewController is UINavigationController { - destinationViewController = originalTopViewController - } - viewController.navigationController?.stp_pop( - to: destinationViewController, - animated: transitionAnimationsEnabled() - ) { - completion() - } - } - } - - // MARK: - Request Payment - func requestPaymentShouldPresentShippingViewController() -> Bool { - let shippingAddressRequired = (configuration.requiredShippingAddressFields?.count ?? 0) > 0 - var shippingAddressIncomplete: Bool? - if let requiredShippingAddressFields1 = configuration.requiredShippingAddressFields { - shippingAddressIncomplete = - !(shippingAddress?.containsRequiredShippingAddressFields( - requiredShippingAddressFields1 - ) - ?? false) - } - let shippingMethodRequired = - configuration.shippingType == .shipping - && delegate?.responds( - to: #selector( - STPPaymentContextDelegate.paymentContext(_:didUpdateShippingAddress:completion:) - ) - ) - ?? false - && selectedShippingMethod == nil - let verificationRequired = - configuration.verifyPrefilledShippingAddress && shippingAddressNeedsVerification - // true if STPShippingVC should be presented to collect or verify a shipping address - let shouldPresentShippingAddress = - shippingAddressRequired && (shippingAddressIncomplete ?? false || verificationRequired) - // this handles a corner case where STPShippingVC should be presented because: - // - shipping address has been pre-filled - // - no verification is required, but the user still needs to enter a shipping method - let shouldPresentShippingMethods = - shippingAddressRequired && !(shippingAddressIncomplete ?? false) - && !verificationRequired - && shippingMethodRequired - return shouldPresentShippingAddress || shouldPresentShippingMethods - } - - func didFinish( - with status: STPPaymentStatus, - error: Error? - ) { - state = STPPaymentContextState.none - delegate?.paymentContext( - self, - didFinishWith: status, - error: error - ) - } - - func buildPaymentRequest() -> PKPaymentRequest? { - guard let appleMerchantIdentifier = configuration.appleMerchantIdentifier, paymentAmount > 0 - else { - return nil - } - let paymentRequest = StripeAPI.paymentRequest( - withMerchantIdentifier: appleMerchantIdentifier, - country: paymentCountry, - currency: paymentCurrency - ) - -#if compiler(>=5.9) - if #available(macOS 14.0, iOS 17.0, *) { - paymentRequest.applePayLaterAvailability = applePayLaterAvailability._convertedToSwiftValue() - } -#endif - - let summaryItems = paymentSummaryItems - paymentRequest.paymentSummaryItems = summaryItems - - let requiredFields = STPAddress.applePayContactFields( - from: configuration.requiredBillingAddressFields - ) - paymentRequest.requiredBillingContactFields = requiredFields - - var shippingRequiredFields: Set? - if let requiredShippingAddressFields1 = configuration.requiredShippingAddressFields { - shippingRequiredFields = STPAddress.pkContactFields( - fromStripeContactFields: requiredShippingAddressFields1 - ) - } - if let shippingRequiredFields = shippingRequiredFields { - paymentRequest.requiredShippingContactFields = shippingRequiredFields - } - - paymentRequest.currencyCode = paymentCurrency.uppercased() - if let selectedShippingMethod = selectedShippingMethod { - var orderedShippingMethods = shippingMethods - orderedShippingMethods?.removeAll { - $0 as AnyObject === selectedShippingMethod as AnyObject - } - orderedShippingMethods?.insert(selectedShippingMethod, at: 0) - paymentRequest.shippingMethods = orderedShippingMethods - } else { - paymentRequest.shippingMethods = shippingMethods - } - - paymentRequest.shippingType = STPPaymentContext.pkShippingType(configuration.shippingType) - - if let shippingAddress = shippingAddress { - paymentRequest.shippingContact = shippingAddress.pkContactValue() - } - return paymentRequest - } - - class func pkShippingType(_ shippingType: STPShippingType) -> PKShippingType { - switch shippingType { - case .shipping: - return .shipping - case .delivery: - return .delivery - @unknown default: - fatalError() - } - } - - func artificiallyRetain(_ host: NSObject) { - objc_setAssociatedObject( - host, - UnsafeRawPointer(&kSTPPaymentCoordinatorAssociatedObjectKey), - self, - .OBJC_ASSOCIATION_RETAIN_NONATOMIC - ) - } - - // MARK: - STPAuthenticationContext - @objc - public func authenticationPresentingViewController() -> UIViewController { - return hostViewController! - } - - @objc - public func prepare(forPresentation completion: @escaping STPVoidBlock) { - if applePayVC != nil && applePayVC?.presentingViewController != nil { - hostViewController?.dismiss( - animated: transitionAnimationsEnabled() - ) { - completion() - } - } else { - completion() - } - } -} - -/// Implement `STPPaymentContextDelegate` to get notified when a payment context changes, finishes, encounters errors, etc. In practice, if your app has a "checkout screen view controller", that is a good candidate to implement this protocol. -@objc public protocol STPPaymentContextDelegate: NSObjectProtocol { - /// Called when the payment context encounters an error when fetching its initial set of data. A few ways to handle this are: - /// - If you're showing the user a checkout page, dismiss the checkout page when this is called and present the error to the user. - /// - Present the error to the user using a `UIAlertController` with two buttons: Retry and Cancel. If they cancel, dismiss your UI. If they Retry, call `retryLoading` on the payment context. - /// To make it harder to get your UI into a bad state, this won't be called until the context's `hostViewController` has finished appearing. - /// - Parameters: - /// - paymentContext: the payment context that encountered the error - /// - error: the error that was encountered - func paymentContext(_ paymentContext: STPPaymentContext, didFailToLoadWithError error: Error) - /// This is called every time the contents of the payment context change. When this is called, you should update your app's UI to reflect the current state of the payment context. For example, if you have a checkout page with a "selected payment method" row, you should update its payment method with `paymentContext.selectedPaymentOption.label`. If that checkout page has a "buy" button, you should enable/disable it depending on the result of `paymentContext.isReadyForPayment`. - /// - Parameter paymentContext: the payment context that changed - func paymentContextDidChange(_ paymentContext: STPPaymentContext) - /// Inside this method, you should make a call to your backend API to make a PaymentIntent with that Customer + payment method, and invoke the `completion` block when that is done. - /// - Parameters: - /// - paymentContext: The context that succeeded - /// - paymentResult: Information associated with the payment that you can pass to your server. You should go to your backend API with this payment result and use the PaymentIntent API to complete the payment. See https://stripe.com/docs/mobile/ios/basic#submit-payment-intents Once that's done call the `completion` block with any error that occurred (or none, if the payment succeeded). - seealso: STPPaymentResult.h - /// - completion: Call this block when you're done creating a payment intent (or subscription, etc) on your backend. If it succeeded, call `completion(STPPaymentStatusSuccess, nil)`. If it failed with an error, call `completion(STPPaymentStatusError, error)`. If the user canceled, call `completion(STPPaymentStatusUserCancellation, nil)`. - func paymentContext( - _ paymentContext: STPPaymentContext, - didCreatePaymentResult paymentResult: STPPaymentResult, - completion: @escaping STPPaymentStatusBlock - ) - /// This is invoked by an `STPPaymentContext` when it is finished. This will be called after the payment is done and all necessary UI has been dismissed. You should inspect the returned `status` and behave appropriately. For example: if it's `STPPaymentStatusSuccess`, show the user a receipt. If it's `STPPaymentStatusError`, inform the user of the error. If it's `STPPaymentStatusUserCancellation`, do nothing. - /// - Parameters: - /// - paymentContext: The payment context that finished - /// - status: The status of the payment - `STPPaymentStatusSuccess` if it succeeded, `STPPaymentStatusError` if it failed with an error (in which case the `error` parameter will be non-nil), `STPPaymentStatusUserCancellation` if the user canceled the payment. - /// - error: An error that occurred, if any. - func paymentContext( - _ paymentContext: STPPaymentContext, - didFinishWith status: STPPaymentStatus, - error: Error? - ) - - /// Inside this method, you should verify that you can ship to the given address. - /// You should call the completion block with the results of your validation - /// and the available shipping methods for the given address. If you don't implement - /// this method, the user won't be prompted to select a shipping method and all - /// addresses will be valid. If you call the completion block with nil or an - /// empty array of shipping methods, the user won't be prompted to select a - /// shipping method. - /// @note If a user updates their shipping address within the Apple Pay dialog, - /// this address will be anonymized. For example, in the US, it will only include the - /// city, state, and zip code. The payment context will have the user's complete - /// shipping address by the time `paymentContext:didFinishWithStatus:error` is - /// called. - /// - Parameters: - /// - paymentContext: The context that updated its shipping address - /// - address: The current shipping address - /// - completion: Call this block when you're done validating the shipping - /// address and calculating available shipping methods. If you call the completion - /// block with nil or an empty array of shipping methods, the user won't be prompted - /// to select a shipping method. - @objc optional func paymentContext( - _ paymentContext: STPPaymentContext, - didUpdateShippingAddress address: STPAddress, - completion: @escaping STPShippingMethodsCompletionBlock - ) -} - -/// The current state of the payment context -/// - STPPaymentContextStateNone: No view controllers are currently being shown. The payment may or may not have already been completed -/// - STPPaymentContextStateShowingRequestedViewController: The view controller that you requested the context show is being shown (via the push or present payment methods or shipping view controller methods) -/// - STPPaymentContextStateRequestingPayment: The payment context is in the middle of requesting payment. It may be showing some other UI or view controller if more information is necessary to complete the payment. -enum STPPaymentContextState: Int { - case none - case showingRequestedViewController - case requestingPayment -} - -private var kSTPPaymentCoordinatorAssociatedObjectKey = 0 - -/// :nodoc: -@_spi(STP) extension STPPaymentContext: STPAnalyticsProtocol { - @_spi(STP) public static var stp_analyticsIdentifier = "STPPaymentContext" -} - -#if compiler(>=5.9) -@available(macOS 14.0, iOS 17.0, *) -extension PKApplePayLaterAvailability { - func _convertedToSwiftValue() -> PKPaymentRequest.ApplePayLaterAvailability { - switch self { - case .available: - return .available - case .unavailableItemIneligible: - return .unavailable(.itemIneligible) - case .unavailableRecurringTransaction: - return .unavailable(.recurringTransaction) - @unknown default: - fatalError() - } - } -} -#endif diff --git a/Stripe/StripeiOS/Source/STPPaymentOptionsInternalViewController.swift b/Stripe/StripeiOS/Source/STPPaymentOptionsInternalViewController.swift deleted file mode 100644 index 10bd0d4848b..00000000000 --- a/Stripe/StripeiOS/Source/STPPaymentOptionsInternalViewController.swift +++ /dev/null @@ -1,593 +0,0 @@ -// -// STPPaymentOptionsInternalViewController.swift -// StripeiOS -// -// Created by Jack Flintermann on 6/9/16. -// Copyright © 2016 Stripe, Inc. All rights reserved. -// - -import Foundation -@_spi(STP) import StripeCore -import UIKit - -@objc protocol STPPaymentOptionsInternalViewControllerDelegate: AnyObject { - func internalViewControllerDidSelect(_ paymentOption: STPPaymentOption?) - func internalViewControllerDidDelete(_ paymentOption: STPPaymentOption?) - func internalViewControllerDidCreatePaymentOption( - _ paymentOption: STPPaymentOption?, - completion: @escaping STPErrorBlock - ) - func internalViewControllerDidCancel() -} - -class STPPaymentOptionsInternalViewController: STPCoreTableViewController, UITableViewDataSource, - UITableViewDelegate, STPAddCardViewControllerDelegate, STPBankSelectionViewControllerDelegate -{ - init( - configuration: STPPaymentConfiguration, - customerContext: STPCustomerContext?, - analyticsLogger: STPPaymentContext.AnalyticsLogger, - apiClient: STPAPIClient, - theme: STPTheme, - prefilledInformation: STPUserInformation?, - shippingAddress: STPAddress?, - paymentOptionTuple tuple: STPPaymentOptionTuple, - delegate: STPPaymentOptionsInternalViewControllerDelegate? - ) { - self.analyticsLogger = analyticsLogger - super.init(theme: theme) - self.configuration = configuration - // This parameter may be a custom API adapter, and not a CustomerContext. - apiAdapter = customerContext - self.apiClient = apiClient - self.prefilledInformation = prefilledInformation - self.shippingAddress = shippingAddress - paymentOptions = tuple.paymentOptions - selectedPaymentOption = tuple.selectedPaymentOption - self.delegate = delegate - - title = STPLocalizedString("Payment Method", "Title for Payment Method screen") - } - - func update(with tuple: STPPaymentOptionTuple) { - if let selectedPaymentOption = selectedPaymentOption, - selectedPaymentOption.isEqual(tuple.selectedPaymentOption) - { - return - } - - paymentOptions = tuple.paymentOptions - selectedPaymentOption = tuple.selectedPaymentOption - - // Reload card list section - let sections = NSMutableIndexSet(index: PaymentOptionSectionCardList) - tableView?.reloadSections(sections as IndexSet, with: .automatic) - } - - private var _customFooterView: UIView? - internal let analyticsLogger: STPPaymentContext.AnalyticsLogger - var customFooterView: UIView? { - get { - _customFooterView - } - set(footerView) { - _customFooterView = footerView - _didSetCustomFooterView() - } - } - func _didSetCustomFooterView() { - if isViewLoaded { - if let size = _customFooterView?.sizeThatFits( - CGSize(width: view.bounds.size.width, height: CGFloat.greatestFiniteMagnitude) - ) { - _customFooterView?.frame = CGRect( - x: 0, - y: 0, - width: size.width, - height: size.height - ) - } - - tableView?.tableFooterView = _customFooterView - } - } - - var addCardViewControllerCustomFooterView: UIView? - var prefilledInformation: STPUserInformation? - private var configuration: STPPaymentConfiguration? - private var apiAdapter: STPBackendAPIAdapter? - private var shippingAddress: STPAddress? - private var paymentOptions: [STPPaymentOption]? - private var apiClient: STPAPIClient = .shared - private var selectedPaymentOption: STPPaymentOption? - private weak var delegate: STPPaymentOptionsInternalViewControllerDelegate? - private var cardImageView: UIImageView? - - override func createAndSetupViews() { - super.createAndSetupViews() - - // Table view - tableView?.register( - STPPaymentOptionTableViewCell.self, - forCellReuseIdentifier: PaymentOptionCellReuseIdentifier - ) - - tableView?.dataSource = self - tableView?.delegate = self - tableView?.reloadData() - - // Table header view - let cardImageView = UIImageView(image: STPLegacyImageLibrary.largeCardFrontImage()) - cardImageView.contentMode = .center - cardImageView.frame = CGRect( - x: 0.0, - y: 0.0, - width: view.bounds.size.width, - height: cardImageView.bounds.size.height + (57.0 * 2.0) - ) - cardImageView.image = STPLegacyImageLibrary.largeCardFrontImage() - cardImageView.tintColor = theme.accentColor - self.cardImageView = cardImageView - - tableView?.tableHeaderView = cardImageView - - // Table view editing state - tableView?.setEditing(false, animated: false) - reloadRightBarButtonItem( - withTableViewIsEditing: tableView?.isEditing ?? false, - animated: false - ) - - stp_navigationItemProxy?.leftBarButtonItem?.accessibilityIdentifier = - "PaymentOptionsViewControllerCancelButtonIdentifier" - // re-set the custom footer view if it was added before we loaded - _didSetCustomFooterView() - } - - override func viewDidAppear(_ animated: Bool) { - super.viewDidAppear(animated) - analyticsLogger.logPaymentOptionsScreenAppeared() - } - - override func viewDidLayoutSubviews() { - super.viewDidLayoutSubviews() - - // Resetting it re-calculates the size based on new view width - // UITableView requires us to call setter again to actually pick up frame - // change on footers - if tableView?.tableFooterView != nil { - customFooterView = tableView?.tableFooterView - } - } - - func reloadRightBarButtonItem(withTableViewIsEditing tableViewIsEditing: Bool, animated: Bool) { - var barButtonItem: UIBarButtonItem? - - if !tableViewIsEditing { - if isAnyPaymentOptionDetachable() { - // Show edit button - barButtonItem = UIBarButtonItem( - barButtonSystemItem: .edit, - target: self, - action: #selector(handleEditButtonTapped(_:)) - ) - } else { - // Show no button - barButtonItem = nil - } - } else { - // Show done button - barButtonItem = UIBarButtonItem( - barButtonSystemItem: .done, - target: self, - action: #selector(handleDoneButtonTapped(_:)) - ) - } - - barButtonItem?.stp_setTheme(theme) - - stp_navigationItemProxy?.setRightBarButton(barButtonItem, animated: animated) - } - - func isAnyPaymentOptionDetachable() -> Bool { - for paymentOption in cardPaymentOptions() { - if isPaymentOptionDetachable(paymentOption) { - return true - } - } - - return false - } - - func isPaymentOptionDetachable(_ paymentOption: STPPaymentOption?) -> Bool { - if !(configuration?.canDeletePaymentOptions ?? false) { - // Feature is disabled - return false - } - - if apiAdapter == nil { - // Cannot detach payment methods without customer context - return false - } - - if !(apiAdapter?.responds( - to: #selector(STPCustomerContext.detachPaymentMethod(fromCustomer:completion:)) - ) - ?? false) - { - // Cannot detach payment methods if customerContext is an apiAdapter - // that doesn't implement detachPaymentMethod - return false - } - - if paymentOption == nil { - // Cannot detach non-existent payment method - return false - } - - if !(paymentOption is STPPaymentMethod) { - // Cannot detach non-payment method - return false - } - - // Payment method can be deleted from customer - return true - } - - func cardPaymentOptions() -> [STPPaymentOption] { - guard let paymentOptions = paymentOptions else { - return [] - } - - return paymentOptions.filter({ (o) -> Bool in - if o is STPPaymentMethodParams { - let paymentMethodParams = o as? STPPaymentMethodParams - if paymentMethodParams?.type != .card { - return false - } - } - return true - }) - } - - func apmPaymentOptions() -> [STPPaymentOption] { - guard let paymentOptions = paymentOptions else { - return [] - } - return paymentOptions.filter({ (o) -> Bool in - if (o) is STPPaymentMethodParams { - let paymentMethodParams = o as? STPPaymentMethodParams - if paymentMethodParams?.type == .FPX { - // Add other APMs as we gain support for them in Basic Integration - return true - } - } - return false - }) - } - - // MARK: - Button Handlers - @objc override func handleCancelTapped(_ sender: Any?) { - delegate?.internalViewControllerDidCancel() - } - - @objc func handleEditButtonTapped(_ sender: Any?) { - tableView?.setEditing(true, animated: true) - reloadRightBarButtonItem( - withTableViewIsEditing: tableView?.isEditing ?? false, - animated: true - ) - } - - @objc func handleDoneButtonTapped(_ sender: Any?) { - _endTableViewEditing() - reloadRightBarButtonItem( - withTableViewIsEditing: tableView?.isEditing ?? false, - animated: true - ) - } - - func _endTableViewEditing() { - tableView?.setEditing(false, animated: true) - } - - // MARK: - UITableViewDataSource - func numberOfSections(in tableView: UITableView) -> Int { - if apmPaymentOptions().count > 0 { - return 3 - } else { - return 2 - } - } - - func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int { - if section == PaymentOptionSectionCardList { - return cardPaymentOptions().count - } - - if section == PaymentOptionSectionAddCard { - return 1 - } - - if section == PaymentOptionSectionAPM { - return apmPaymentOptions().count - } - - return 0 - } - - func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell { - let cell = - tableView.dequeueReusableCell( - withIdentifier: PaymentOptionCellReuseIdentifier, - for: indexPath - ) - as? STPPaymentOptionTableViewCell - - if indexPath.section == PaymentOptionSectionCardList { - weak var paymentOption = - cardPaymentOptions().stp_boundSafeObject(at: indexPath.row) - let selected = paymentOption!.isEqual(selectedPaymentOption) - - cell?.configure(with: paymentOption!, theme: theme, selected: selected) - } else if indexPath.section == PaymentOptionSectionAddCard { - cell?.configureForNewCardRow(with: theme) - cell?.accessibilityIdentifier = "PaymentOptionsTableViewAddNewCardButtonIdentifier" - } else if indexPath.section == PaymentOptionSectionAPM { - weak var paymentOption = - apmPaymentOptions().stp_boundSafeObject(at: indexPath.row) - if paymentOption is STPPaymentMethodParams { - let paymentMethodParams = paymentOption as? STPPaymentMethodParams - if paymentMethodParams?.type == .FPX { - cell?.configureForFPXRow(with: theme) - cell?.accessibilityIdentifier = "PaymentOptionsTableViewFPXButtonIdentifier" - } - } - } - - return cell! - } - - func tableView(_ tableView: UITableView, canEditRowAt indexPath: IndexPath) -> Bool { - if indexPath.section == PaymentOptionSectionCardList { - weak var paymentOption = - cardPaymentOptions().stp_boundSafeObject(at: indexPath.row) - - if isPaymentOptionDetachable(paymentOption) { - return true - } - } - - return false - } - - func tableView( - _ tableView: UITableView, - commit editingStyle: UITableViewCell.EditingStyle, - forRowAt indexPath: IndexPath - ) { - if indexPath.section == PaymentOptionSectionCardList { - if editingStyle != .delete { - // Showed the user a non-delete option when we shouldn't have - tableView.reloadData() - return - } - - if !(indexPath.row < cardPaymentOptions().count) { - // Data source and table view out of sync for some reason - tableView.reloadData() - return - } - - weak var paymentOptionToDelete = - cardPaymentOptions().stp_boundSafeObject(at: indexPath.row) - - if !isPaymentOptionDetachable(paymentOptionToDelete) { - // Showed the user a delete option for a payment method when we shouldn't have - tableView.reloadData() - return - } - - let paymentMethod = paymentOptionToDelete as? STPPaymentMethod - - // Kickoff request to delete payment method from customer - if let paymentMethod = paymentMethod { - apiAdapter?.detachPaymentMethod?(fromCustomer: paymentMethod, completion: nil) - } - - // Optimistically remove payment method from data source - var paymentOptions = self.paymentOptions - paymentOptions?.removeAll { $0 as AnyObject === paymentOptionToDelete as AnyObject } - self.paymentOptions = paymentOptions - - // Perform deletion animation for single row - tableView.deleteRows(at: [indexPath], with: .automatic) - - var tableViewIsEditing = tableView.isEditing - if !isAnyPaymentOptionDetachable() { - // we deleted the last available payment option, stop editing - // (but delay to next runloop because calling tableView setEditing:animated: - // in this function is not allowed) - DispatchQueue.main.async(execute: { - self._endTableViewEditing() - }) - // manually set the value passed to reloadRightBarButtonItemWithTableViewIsEditing - // below - tableViewIsEditing = false - } - - // Reload right bar button item text - reloadRightBarButtonItem(withTableViewIsEditing: tableViewIsEditing, animated: true) - - // Notify delegate - delegate?.internalViewControllerDidDelete(paymentOptionToDelete) - } - } - - // MARK: - UITableViewDelegate - func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) { - if indexPath.section == PaymentOptionSectionCardList { - // Update data source - weak var paymentOption = - cardPaymentOptions().stp_boundSafeObject(at: indexPath.row) - selectedPaymentOption = paymentOption - - // Perform selection animation - tableView.reloadSections( - NSIndexSet(index: PaymentOptionSectionCardList) as IndexSet, - with: .fade - ) - - // Notify delegate - delegate?.internalViewControllerDidSelect(paymentOption) - } else if indexPath.section == PaymentOptionSectionAddCard { - var paymentCardViewController: STPAddCardViewController? - if let configuration = configuration { - paymentCardViewController = STPAddCardViewController( - configuration: configuration, - theme: theme - ) - } - paymentCardViewController?.analyticsLogger = analyticsLogger - paymentCardViewController?.apiClient = apiClient - paymentCardViewController?.delegate = self - paymentCardViewController?.prefilledInformation = prefilledInformation - paymentCardViewController?.shippingAddress = shippingAddress - paymentCardViewController?.customFooterView = addCardViewControllerCustomFooterView - - if let paymentCardViewController = paymentCardViewController { - navigationController?.pushViewController(paymentCardViewController, animated: true) - } - } else if indexPath.section == PaymentOptionSectionAPM { - weak var paymentOption = - apmPaymentOptions().stp_boundSafeObject(at: indexPath.row) - if paymentOption is STPPaymentMethodParams { - if let paymentMethodParams = paymentOption as? STPPaymentMethodParams, - paymentMethodParams.type == .FPX - { - var bankSelectionViewController: STPBankSelectionViewController? - if let configuration = configuration { - bankSelectionViewController = STPBankSelectionViewController( - bankMethod: .FPX, - configuration: configuration, - theme: theme - ) - } - bankSelectionViewController?.apiClient = apiClient - bankSelectionViewController?.delegate = self - - if let bankSelectionViewController = bankSelectionViewController { - navigationController?.pushViewController( - bankSelectionViewController, - animated: true - ) - } - } - } - } - - tableView.deselectRow(at: indexPath, animated: true) - } - - func tableView( - _ tableView: UITableView, - willDisplay cell: UITableViewCell, - forRowAt indexPath: IndexPath - ) { - let isTopRow = indexPath.row == 0 - let isBottomRow = - self.tableView(tableView, numberOfRowsInSection: indexPath.section) - 1 == indexPath.row - - cell.stp_setBorderColor(theme.tertiaryBackgroundColor) - cell.stp_setTopBorderHidden(!isTopRow) - cell.stp_setBottomBorderHidden(!isBottomRow) - cell.stp_setFakeSeparatorColor(theme.quaternaryBackgroundColor) - cell.stp_setFakeSeparatorLeftInset(15.0) - } - - func tableView(_ tableView: UITableView, heightForFooterInSection section: Int) -> CGFloat { - if self.tableView(tableView, numberOfRowsInSection: section) == 0 { - return 0.01 - } - - return 27.0 - } - - override func tableView( - _ tableView: UITableView, - heightForHeaderInSection section: Int - ) - -> CGFloat - { - return 0.01 - } - - func tableView( - _ tableView: UITableView, - editingStyleForRowAt indexPath: IndexPath - ) - -> UITableViewCell.EditingStyle - { - if indexPath.section == PaymentOptionSectionCardList { - return .delete - } - - return .none - } - - func tableView(_ tableView: UITableView, willBeginEditingRowAt indexPath: IndexPath) { - reloadRightBarButtonItem(withTableViewIsEditing: true, animated: true) - } - - func tableView(_ tableView: UITableView, didEndEditingRowAt indexPath: IndexPath?) { - reloadRightBarButtonItem(withTableViewIsEditing: tableView.isEditing, animated: true) - } - - // MARK: - STPAddCardViewControllerDelegate - func addCardViewControllerDidCancel(_ addCardViewController: STPAddCardViewController) { - navigationController?.popViewController(animated: true) - } - - @objc func addCardViewController( - _ addCardViewController: STPAddCardViewController, - didCreatePaymentMethod paymentMethod: STPPaymentMethod, - completion: @escaping STPErrorBlock - ) { - delegate?.internalViewControllerDidCreatePaymentOption( - paymentMethod, - completion: completion - ) - } - - @objc func bankSelectionViewController( - _ bankViewController: STPBankSelectionViewController, - didCreatePaymentMethodParams paymentMethodParams: STPPaymentMethodParams - ) { - delegate?.internalViewControllerDidCreatePaymentOption(paymentMethodParams) { _ in - } - } - - required init?( - coder aDecoder: NSCoder - ) { - fatalError("init(coder:) has not been implemented") - } - - required init( - nibName nibNameOrNil: String?, - bundle nibBundleOrNil: Bundle? - ) { - fatalError("init(nibName:bundle:) has not been implemented") - } - - required init( - theme: STPTheme? - ) { - fatalError("init(theme:) has not been implemented") - } -} - -private let PaymentOptionCellReuseIdentifier = "PaymentOptionCellReuseIdentifier" -private let PaymentOptionSectionCardList = 0 -private let PaymentOptionSectionAddCard = 1 -private let PaymentOptionSectionAPM = 2 diff --git a/Stripe/StripeiOS/Source/STPPaymentOptionsViewController.swift b/Stripe/StripeiOS/Source/STPPaymentOptionsViewController.swift deleted file mode 100644 index aa392eae4cc..00000000000 --- a/Stripe/StripeiOS/Source/STPPaymentOptionsViewController.swift +++ /dev/null @@ -1,662 +0,0 @@ -// -// STPPaymentOptionsViewController.swift -// StripeiOS -// -// Created by Jack Flintermann on 1/12/16. -// Copyright © 2016 Stripe, Inc. All rights reserved. -// - -@_spi(STP) import StripeCore -@_spi(STP) import StripePaymentsUI -import UIKit - -/// This view controller presents a list of payment method options to the user, -/// which they can select between. They can also add credit cards to the list. -/// It must be displayed inside a `UINavigationController`, so you can either -/// create a `UINavigationController` with an `STPPaymentOptionsViewController` -/// as the `rootViewController` and then present the `UINavigationController`, -/// or push a new `STPPaymentOptionsViewController` onto an existing -/// `UINavigationController`'s stack. You can also have `STPPaymentContext` do this -/// for you automatically, by calling `presentPaymentOptionsViewController` -/// or `pushPaymentOptionsViewController` on it. -public class STPPaymentOptionsViewController: STPCoreViewController, - STPPaymentOptionsInternalViewControllerDelegate, STPAddCardViewControllerDelegate -{ - - /// The delegate for the view controller. - /// The delegate receives callbacks when the user selects a method or cancels, - /// and is responsible for dismissing the payments methods view controller when - /// it is finished. - @objc private(set) weak var delegate: STPPaymentOptionsViewControllerDelegate? - - /// Creates a new payment methods view controller. - /// - Parameter paymentContext: A payment context to power the view controller's view. - /// The payment context will in turn use its backend API adapter to fetch the - /// information it needs from your application. - /// - Returns: an initialized view controller. - @objc(initWithPaymentContext:) - public convenience init( - paymentContext: STPPaymentContext - ) { - self.init( - configuration: paymentContext.configuration, - apiAdapter: paymentContext.apiAdapter, - apiClient: paymentContext.apiClient, - analyticsLogger: paymentContext.analyticsLogger, - loadingPromise: paymentContext.currentValuePromise, - theme: paymentContext.theme, - shippingAddress: paymentContext.shippingAddress, - delegate: paymentContext - ) - } - - init( - configuration: STPPaymentConfiguration?, - apiAdapter: STPBackendAPIAdapter, - apiClient: STPAPIClient?, - analyticsLogger: STPPaymentContext.AnalyticsLogger, - loadingPromise: STPPromise?, - theme: STPTheme?, - shippingAddress: STPAddress?, - delegate: STPPaymentOptionsViewControllerDelegate - ) { - self.apiAdapter = apiAdapter - self.analyticsLogger = analyticsLogger - super.init(theme: theme) - commonInit( - configuration: configuration, - apiAdapter: apiAdapter, - apiClient: apiClient, - loadingPromise: loadingPromise, - shippingAddress: shippingAddress, - delegate: delegate - ) - } - - func commonInit( - configuration: STPPaymentConfiguration?, - apiAdapter: STPBackendAPIAdapter, - apiClient: STPAPIClient?, - loadingPromise: STPPromise?, - shippingAddress: STPAddress?, - delegate: STPPaymentOptionsViewControllerDelegate - ) { - STPAnalyticsClient.sharedClient.addClass( - toProductUsageIfNecessary: STPPaymentOptionsViewController.self - ) - - self.configuration = configuration - self.apiClient = apiClient ?? .shared - self.shippingAddress = shippingAddress - self.apiAdapter = apiAdapter - self.loadingPromise = loadingPromise - self.delegate = delegate - - navigationItem.title = STPLocalizedString( - "Loading…", - "Title for screen when data is still loading from the network." - ) - - weak var weakSelf = self - loadingPromise?.onSuccess({ tuple in - guard let strongSelf = weakSelf else { - return - } - var `internal`: UIViewController? - if (tuple.paymentOptions.count) > 0 { - let customerContext = strongSelf.apiAdapter as? STPCustomerContext - - var payMethodsInternal: STPPaymentOptionsInternalViewController? - if let configuration1 = strongSelf.configuration { - payMethodsInternal = STPPaymentOptionsInternalViewController( - configuration: configuration1, - customerContext: customerContext, - analyticsLogger: strongSelf.analyticsLogger, - apiClient: strongSelf.apiClient, - theme: strongSelf.theme, - prefilledInformation: strongSelf.prefilledInformation, - shippingAddress: strongSelf.shippingAddress, - paymentOptionTuple: tuple, - delegate: strongSelf - ) - } - if strongSelf.paymentOptionsViewControllerFooterView != nil { - payMethodsInternal?.customFooterView = - strongSelf.paymentOptionsViewControllerFooterView - } - if strongSelf.addCardViewControllerFooterView != nil { - payMethodsInternal?.addCardViewControllerCustomFooterView = - strongSelf.addCardViewControllerFooterView - } - `internal` = payMethodsInternal - } else { - var addCardViewController: STPAddCardViewController? - if let configuration1 = strongSelf.configuration { - addCardViewController = STPAddCardViewController( - configuration: configuration1, - theme: strongSelf.theme - ) - } - addCardViewController?.analyticsLogger = strongSelf.analyticsLogger - addCardViewController?.apiClient = strongSelf.apiClient - addCardViewController?.delegate = strongSelf - addCardViewController?.prefilledInformation = strongSelf.prefilledInformation - addCardViewController?.shippingAddress = strongSelf.shippingAddress - `internal` = addCardViewController - - if strongSelf.addCardViewControllerFooterView != nil { - addCardViewController?.customFooterView = - strongSelf.addCardViewControllerFooterView - } - } - - `internal`?.stp_navigationItemProxy = strongSelf.navigationItem - if let controller = `internal` { - strongSelf.addChild(controller) - } - `internal`?.view.alpha = 0 - if let view = `internal`?.view, let activityIndicator1 = strongSelf.activityIndicator { - strongSelf.view.insertSubview(view, belowSubview: activityIndicator1) - } - if let view = `internal`?.view { - strongSelf.view.addSubview(view) - } - `internal`?.view.frame = strongSelf.view.bounds - `internal`?.didMove(toParent: strongSelf) - UIView.animate( - withDuration: 0.2, - animations: { - strongSelf.activityIndicator?.alpha = 0 - `internal`?.view.alpha = 1 - } - ) { _ in - strongSelf.activityIndicator?.animating = false - } - strongSelf.navigationItem.setRightBarButton( - `internal`?.stp_navigationItemProxy?.rightBarButtonItem, - animated: true - ) - strongSelf.internalViewController = `internal` - }) - } - - /// Initializes a new payment methods view controller without using a - /// payment context. - /// - Parameters: - /// - configuration: The configuration to use to determine what types of - /// payment method to offer your user. - seealso: STPPaymentConfiguration.h - /// - theme: The theme to inform the appearance of the UI. - /// - customerContext: The customer context the view controller will use to - /// fetch and modify its Stripe customer - /// - delegate: A delegate that will be notified when the payment - /// methods view controller's selection changes. - /// - Returns: an initialized view controller. - @objc(initWithConfiguration:theme:customerContext:delegate:) - public convenience init( - configuration: STPPaymentConfiguration, - theme: STPTheme, - customerContext: STPCustomerContext, - delegate: STPPaymentOptionsViewControllerDelegate - ) { - self.init( - configuration: configuration, - theme: theme, - apiAdapter: customerContext, - delegate: delegate - ) - } - - /// Note: Instead of providing your own backend API adapter, we recommend using - /// `STPCustomerContext`, which will manage retrieving and updating a - /// Stripe customer for you. - seealso: STPCustomerContext.h - /// Initializes a new payment methods view controller without using - /// a payment context. - /// - Parameters: - /// - configuration: The configuration to use to determine what types of - /// payment method to offer your user. - /// - theme: The theme to inform the appearance of the UI. - /// - apiAdapter: The API adapter to use to retrieve a customer's stored - /// payment methods and save new ones. - /// - delegate: A delegate that will be notified when the payment methods - /// view controller's selection changes. - @objc(initWithConfiguration:theme:apiAdapter:delegate:) - public init( - configuration: STPPaymentConfiguration, - theme: STPTheme, - apiAdapter: STPBackendAPIAdapter, - delegate: STPPaymentOptionsViewControllerDelegate - ) { - self.apiAdapter = apiAdapter - super.init(theme: theme) - let promise = retrievePaymentMethods(with: configuration, apiAdapter: apiAdapter) - - commonInit( - configuration: configuration, - apiAdapter: apiAdapter, - apiClient: STPAPIClient.shared, - loadingPromise: promise, - shippingAddress: nil, - delegate: delegate - ) - } - - /// If you've already collected some information from your user, you can set it - /// here and it'll be automatically filled out when possible/appropriate in any UI - /// that the payment context creates. - @objc public var prefilledInformation: STPUserInformation? { - didSet { - if let payMethodsInternal = internalViewController as? STPPaymentOptionsInternalViewController { - payMethodsInternal.prefilledInformation = prefilledInformation - } else if let payMethodsInternal = internalViewController as? STPAddCardViewController { - payMethodsInternal.prefilledInformation = prefilledInformation - } - } - } - /// @note This is no longer recommended as of v18.3.0 - the SDK automatically saves the Stripe ID of the last selected - /// payment method using NSUserDefaults and displays it as the default pre-selected option. You can override this behavior - /// by setting this property. - /// The Stripe ID of a payment method to display as the default pre-selected option. - /// @note Setting this after the view controller's view has loaded has no effect. - @objc public var defaultPaymentMethod: String? - /// A view that will be placed as the footer of the view controller when it is - /// showing a list of saved payment methods to select from. - /// When the footer view needs to be resized, it will be sent a - /// `sizeThatFits:` call. The view should respond correctly to this method in order - /// to be sized and positioned properly. - @objc public var paymentOptionsViewControllerFooterView: UIView? { - didSet { - if let payMethodsInternal = internalViewController as? STPPaymentOptionsInternalViewController { - payMethodsInternal.customFooterView = paymentOptionsViewControllerFooterView - } - } - } - - /// A view that will be placed as the footer of the view controller when it is - /// showing the add card view. - /// When the footer view needs to be resized, it will be sent a - /// `sizeThatFits:` call. The view should respond correctly to this method in order - /// to be sized and positioned properly. - @objc public var addCardViewControllerFooterView: UIView? { - didSet { - if let payMethodsInternal = internalViewController as? STPPaymentOptionsInternalViewController { - payMethodsInternal.addCardViewControllerCustomFooterView = addCardViewControllerFooterView - } else if let payMethodsInternal = internalViewController as? STPAddCardViewController { - payMethodsInternal.customFooterView = addCardViewControllerFooterView - } - } - } - - /// The API Client to use to make requests. - /// Defaults to STPAPIClient.shared - public var apiClient: STPAPIClient = .shared - - /// If you're pushing `STPPaymentOptionsViewController` onto an existing - /// `UINavigationController`'s stack, you should use this method to dismiss it, - /// since it may have pushed an additional add card view controller onto the - /// navigation controller's stack. - /// - Parameter completion: The callback to run after the view controller is dismissed. - /// You may specify nil for this parameter. - @objc(dismissWithCompletion:) - public func dismiss(withCompletion completion: STPVoidBlock?) { - if stp_isAtRootOfNavigationController() { - presentingViewController?.dismiss(animated: true, completion: completion) - } else { - var previous = navigationController?.viewControllers.first - for viewController in navigationController?.viewControllers ?? [] { - if viewController == self { - break - } - previous = viewController - } - navigationController?.stp_pop( - to: previous, - animated: true, - completion: completion ?? {} - ) - } - } - - /// Use one of the initializers declared in this interface. - @available( - *, - unavailable, - message: "Use one of the initializers declared in this interface instead." - ) - @objc public required init( - theme: STPTheme? - ) { - fatalError("init(theme:) has not been implemented") - } - - /// Use one of the initializers declared in this interface. - @available( - *, - unavailable, - message: "Use one of the initializers declared in this interface instead." - ) - @objc public required init( - nibName nibNameOrNil: String?, - bundle nibBundleOrNil: Bundle? - ) { - fatalError("init(nibName:bundle:) has not been implemented") - } - - /// Use one of the initializers declared in this interface. - @available( - *, - unavailable, - message: "Use one of the initializers declared in this interface instead." - ) - @objc public required init?( - coder aDecoder: NSCoder - ) { - fatalError("init(coder:) has not been implemented") - } - - private var configuration: STPPaymentConfiguration? - private var shippingAddress: STPAddress? - private var apiAdapter: STPBackendAPIAdapter - var loadingPromise: STPPromise? - private var activityIndicator: STPPaymentActivityIndicatorView? - internal var internalViewController: UIViewController? - // Should be overwritten if this class is used by STPPaymentContext - internal var analyticsLogger: STPPaymentContext.AnalyticsLogger = .init(product: STPPaymentOptionsViewController.self) - - func retrievePaymentMethods( - with configuration: STPPaymentConfiguration, - apiAdapter: STPBackendAPIAdapter? - ) -> STPPromise { - let promise = STPPromise() - apiAdapter?.listPaymentMethodsForCustomer(completion: { paymentMethods, error in - // We don't use stpDispatchToMainThreadIfNecessary here because we want this completion block to always be called asynchronously, so that users can set self.defaultPaymentMethod in time. - DispatchQueue.main.async(execute: { - if let error = error { - promise.fail(error) - } else { - let defaultPaymentMethod = self.defaultPaymentMethod - if defaultPaymentMethod == nil && (apiAdapter is STPCustomerContext) { - // Retrieve the last selected payment method saved by STPCustomerContext - (apiAdapter as? STPCustomerContext)? - .retrieveLastSelectedPaymentMethodIDForCustomer( - completion: { paymentMethodID, _ in - var paymentTuple: STPPaymentOptionTuple? - if let paymentMethods = paymentMethods { - paymentTuple = STPPaymentOptionTuple.init( - filteredForUIWith: paymentMethods, - selectedPaymentMethod: paymentMethodID, - configuration: configuration - ) - } - promise.succeed(paymentTuple!) - }) - } - var paymentTuple: STPPaymentOptionTuple? - if let paymentMethods = paymentMethods { - paymentTuple = STPPaymentOptionTuple.init( - filteredForUIWith: paymentMethods, - selectedPaymentMethod: defaultPaymentMethod, - configuration: configuration - ) - } - promise.succeed(paymentTuple!) - } - }) - }) - return promise - } - - override func createAndSetupViews() { - super.createAndSetupViews() - - let activityIndicator = STPPaymentActivityIndicatorView() - activityIndicator.animating = true - view.addSubview(activityIndicator) - self.activityIndicator = activityIndicator - } - - /// :nodoc: - @objc - public override func viewDidLayoutSubviews() { - super.viewDidLayoutSubviews() - let centerX = (view.frame.size.width - (activityIndicator?.frame.size.width ?? 0.0)) / 2 - let centerY = (view.frame.size.height - (activityIndicator?.frame.size.height ?? 0.0)) / 2 - activityIndicator?.frame = CGRect( - x: centerX, - y: centerY, - width: activityIndicator?.frame.size.width ?? 0.0, - height: activityIndicator?.frame.size.height ?? 0.0 - ) - internalViewController?.view.frame = view.bounds - } - - /// :nodoc: - @objc - public override func viewDidAppear(_ animated: Bool) { - super.viewDidAppear(animated) - weak var weakSelf = self - loadingPromise?.onSuccess({ tuple in - let strongSelf = weakSelf - if strongSelf == nil { - return - } - - if tuple.selectedPaymentOption != nil { - if strongSelf?.delegate?.responds( - to: #selector( - STPPaymentOptionsViewControllerDelegate.paymentOptionsViewController( - _: - didSelect: - )) - ) - ?? false - { - if let strongSelf = strongSelf, - let selectedPaymentOption = tuple.selectedPaymentOption - { - strongSelf.delegate?.paymentOptionsViewController?( - strongSelf, - didSelect: selectedPaymentOption - ) - } - } - } - }).onFailure({ error in - let strongSelf = weakSelf - if strongSelf == nil { - return - } - - if let strongSelf = strongSelf { - strongSelf.delegate?.paymentOptionsViewController( - strongSelf, - didFailToLoadWithError: error - ) - } - }) - } - - @objc override func updateAppearance() { - super.updateAppearance() - - activityIndicator?.tintColor = theme.accentColor - } - - func finish(with paymentOption: STPPaymentOption?) { - let isReusablePaymentMethod = - (paymentOption is STPPaymentMethod) - && (paymentOption as? STPPaymentMethod)?.isReusable ?? false - - if apiAdapter is STPCustomerContext { - if isReusablePaymentMethod { - // Save the payment method - let paymentMethod = paymentOption as? STPPaymentMethod - (apiAdapter as? STPCustomerContext)?.saveLastSelectedPaymentMethodID( - forCustomer: paymentMethod?.stripeId ?? "", - completion: nil - ) - } else { - // The customer selected something else (like Apple Pay) - (apiAdapter as? STPCustomerContext)?.saveLastSelectedPaymentMethodID( - forCustomer: nil, - completion: nil - ) - } - } - - if delegate?.responds( - to: #selector( - STPPaymentOptionsViewControllerDelegate.paymentOptionsViewController(_:didSelect:)) - ) - ?? false - { - if let paymentOption = paymentOption { - delegate?.paymentOptionsViewController?(self, didSelect: paymentOption) - } - } - delegate?.paymentOptionsViewControllerDidFinish(self) - } - - func internalViewControllerDidSelect(_ paymentOption: STPPaymentOption?) { - finish(with: paymentOption) - } - - func internalViewControllerDidDelete(_ paymentOption: STPPaymentOption?) { - if delegate is STPPaymentContext { - // Notify payment context to update its copy of payment methods - if let paymentContext = delegate as? STPPaymentContext, - let paymentOption = paymentOption - { - paymentContext.remove(paymentOption) - } - } - } - - func internalViewControllerDidCreatePaymentOption( - _ paymentOption: STPPaymentOption?, - completion: @escaping STPErrorBlock - ) { - if !(paymentOption?.isReusable ?? false) { - // Don't save a non-reusable payment option - finish(with: paymentOption) - return - } - let paymentMethod = paymentOption as? STPPaymentMethod - if let paymentMethod = paymentMethod { - apiAdapter.attachPaymentMethod(toCustomer: paymentMethod) { error in - stpDispatchToMainThreadIfNecessary({ - completion(error) - if error == nil { - var promise: STPPromise? - if let configuration = self.configuration { - promise = self.retrievePaymentMethods( - with: configuration, - apiAdapter: self.apiAdapter - ) - } - weak var weakSelf = self - promise?.onSuccess({ tuple in - let strongSelf = weakSelf - if strongSelf == nil { - return - } - let paymentTuple = STPPaymentOptionTuple( - paymentOptions: tuple.paymentOptions, - selectedPaymentOption: paymentMethod - ) - if strongSelf?.internalViewController - is STPPaymentOptionsInternalViewController - { - let paymentOptionsVC = - strongSelf?.internalViewController - as? STPPaymentOptionsInternalViewController - paymentOptionsVC?.update(with: paymentTuple) - } - }) - self.finish(with: paymentMethod) - } - }) - } - } - } - - func internalViewControllerDidCancel() { - delegate?.paymentOptionsViewControllerDidCancel(self) - } - - @objc override func handleCancelTapped(_ sender: Any?) { - delegate?.paymentOptionsViewControllerDidCancel(self) - } - - @objc - public func addCardViewControllerDidCancel( - _ addCardViewController: STPAddCardViewController - ) { - // Add card is only our direct delegate if there are no other payment methods possible - // and we skipped directly to this screen. In this case, a cancel from it is the same as a cancel to us. - delegate?.paymentOptionsViewControllerDidCancel(self) - } - - @objc - public func addCardViewController( - _ addCardViewController: STPAddCardViewController, - didCreatePaymentMethod paymentMethod: STPPaymentMethod, - completion: @escaping STPErrorBlock - ) { - internalViewControllerDidCreatePaymentOption(paymentMethod, completion: completion) - } -} - -// MARK: - STPPaymentOptionsViewControllerDelegate - -/// An `STPPaymentOptionsViewControllerDelegate` responds when a user selects a -/// payment option from (or cancels) an `STPPaymentOptionsViewController`. In both -/// of these instances, you should dismiss the view controller (either by popping -/// it off the navigation stack, or dismissing it). -@objc public protocol STPPaymentOptionsViewControllerDelegate: NSObjectProtocol { - /// This is called when the view controller encounters an error fetching the user's - /// payment options from its API adapter. You should dismiss the view controller - /// when this is called. - /// - Parameters: - /// - paymentOptionsViewController: the view controller in question - /// - error: the error that occurred - func paymentOptionsViewController( - _ paymentOptionsViewController: STPPaymentOptionsViewController, - didFailToLoadWithError error: Error - ) - /// This is called when the user selects or adds a payment method, so it will often - /// be called immediately after calling `paymentOptionsViewController:didSelectPaymentOption:`. - /// You should dismiss the view controller when this is called. - /// - Parameter paymentOptionsViewController: the view controller that has finished - func paymentOptionsViewControllerDidFinish( - _ paymentOptionsViewController: STPPaymentOptionsViewController - ) - /// This is called when the user taps "cancel". - /// You should dismiss the view controller when this is called. - /// - Parameter paymentOptionsViewController: the view controller that has finished - func paymentOptionsViewControllerDidCancel( - _ paymentOptionsViewController: STPPaymentOptionsViewController - ) - - /// This is called when the user either makes a selection, or adds a new card. - /// This will be triggered after the view controller loads with the user's current - /// selection (if they have one) and then subsequently when they change their - /// choice. You should use this callback to update any necessary UI in your app - /// that displays the user's currently selected payment method. You should *not* - /// dismiss the view controller at this point, instead do this in - /// `paymentOptionsViewControllerDidFinish:`. `STPPaymentOptionsViewController` - /// will also call the necessary methods on your API adapter, so you don't need to - /// call them directly during this method. - /// - Parameters: - /// - paymentOptionsViewController: the view controller in question - /// - paymentOption: the selected payment method - @objc(paymentOptionsViewController:didSelectPaymentOption:) - optional func paymentOptionsViewController( - _ paymentOptionsViewController: STPPaymentOptionsViewController, - didSelect paymentOption: STPPaymentOption - ) -} - -/// :nodoc: -@_spi(STP) extension STPPaymentOptionsViewController: STPAnalyticsProtocol { - @_spi(STP) public static var stp_analyticsIdentifier = "STPPaymentOptionsViewController" -} diff --git a/Stripe/StripeiOS/Source/STPShippingAddressViewController.swift b/Stripe/StripeiOS/Source/STPShippingAddressViewController.swift deleted file mode 100644 index ac63189dcee..00000000000 --- a/Stripe/StripeiOS/Source/STPShippingAddressViewController.swift +++ /dev/null @@ -1,674 +0,0 @@ -// -// STPShippingAddressViewController.swift -// StripeiOS -// -// Created by Ben Guo on 8/29/16. -// Copyright © 2016 Stripe, Inc. All rights reserved. -// - -import PassKit -@_spi(STP) import StripeCore -@_spi(STP) import StripePaymentsUI -@_spi(STP) import StripeUICore -import UIKit - -/// This view controller contains a shipping address collection form. It renders a right bar button item that submits the form, so it must be shown inside a `UINavigationController`. Depending on your configuration's shippingType, the view controller may present a shipping method selection form after the user enters an address. -public class STPShippingAddressViewController: STPCoreTableViewController { - - /// A convenience initializer; equivalent to calling `init(configuration: STPPaymentConfiguration.shared theme: STPTheme.defaultTheme currency:"" shippingAddress:nil selectedShippingMethod:nil prefilledInformation:nil)`. - @objc - public convenience init() { - self.init( - configuration: STPPaymentConfiguration.shared, - theme: STPTheme.defaultTheme, - currency: "", - shippingAddress: nil, - selectedShippingMethod: nil, - prefilledInformation: nil - ) - } - - /// Initializes a new `STPShippingAddressViewController` with the given payment context and sets the payment context as its delegate. - /// - Parameter paymentContext: The payment context to use. - @objc(initWithPaymentContext:) - public convenience init( - paymentContext: STPPaymentContext - ) { - STPAnalyticsClient.sharedClient.addClass( - toProductUsageIfNecessary: STPShippingAddressViewController.self - ) - - var billingAddress: STPAddress? - weak var paymentOption = paymentContext.selectedPaymentOption - if paymentOption is STPCard { - let card = paymentOption as? STPCard - billingAddress = card?.address - } else if paymentOption is STPPaymentMethod { - let paymentMethod = paymentOption as? STPPaymentMethod - if let billingDetails1 = paymentMethod?.billingDetails { - billingAddress = STPAddress(paymentMethodBillingDetails: billingDetails1) - } - } - var prefilledInformation: STPUserInformation? - if paymentContext.prefilledInformation != nil { - prefilledInformation = paymentContext.prefilledInformation - } else { - prefilledInformation = STPUserInformation() - } - prefilledInformation?.billingAddress = billingAddress - self.init( - configuration: paymentContext.configuration, - theme: paymentContext.theme, - currency: paymentContext.paymentCurrency, - shippingAddress: paymentContext.shippingAddress, - selectedShippingMethod: paymentContext.selectedShippingMethod, - prefilledInformation: prefilledInformation - ) - - self.delegate = paymentContext - } - - /// Initializes a new `STPShippingAddressCardViewController` with the provided parameters. - /// - Parameters: - /// - configuration: The configuration to use (this determines the required shipping address fields and shipping type). - seealso: STPPaymentConfiguration - /// - theme: The theme to use to inform the view controller's visual appearance. - seealso: STPTheme - /// - currency: The currency to use when displaying amounts for shipping methods. The default is USD. - /// - shippingAddress: If set, the shipping address view controller will be pre-filled with this address. - seealso: STPAddress - /// - selectedShippingMethod: If set, the shipping methods view controller will use this method as the selected shipping method. If `selectedShippingMethod` is nil, the first shipping method in the array of methods returned by your delegate will be selected. - /// - prefilledInformation: If set, the shipping address view controller will be pre-filled with this information. - seealso: STPUserInformation - @objc( - initWithConfiguration: - theme: - currency: - shippingAddress: - selectedShippingMethod: - prefilledInformation: - ) - public init( - configuration: STPPaymentConfiguration, - theme: STPTheme, - currency: String?, - shippingAddress: STPAddress?, - selectedShippingMethod: PKShippingMethod?, - prefilledInformation: STPUserInformation? - ) { - STPAnalyticsClient.sharedClient.addClass( - toProductUsageIfNecessary: STPShippingAddressViewController.self - ) - addressViewModel = STPAddressViewModel( - requiredBillingFields: configuration.requiredBillingAddressFields, - availableCountries: configuration.availableCountries - ) - super.init(theme: theme) - assert( - (configuration.requiredShippingAddressFields?.count ?? 0) > 0, - "`requiredShippingAddressFields` must not be empty when initializing an STPShippingAddressViewController." - ) - self.configuration = configuration - self.currency = currency - self.selectedShippingMethod = selectedShippingMethod - billingAddress = prefilledInformation?.billingAddress - hasUsedBillingAddress = false - addressViewModel = STPAddressViewModel( - requiredShippingFields: configuration.requiredShippingAddressFields ?? [], - availableCountries: configuration.availableCountries - ) - addressViewModel.delegate = self - if let shippingAddress = shippingAddress { - addressViewModel.address = shippingAddress - } else if prefilledInformation?.shippingAddress != nil { - addressViewModel.address = prefilledInformation?.shippingAddress ?? STPAddress() - } - title = title(for: self.configuration?.shippingType ?? .shipping) - } - - /// The view controller's delegate. This must be set before showing the view controller in order for it to work properly. - seealso: STPShippingAddressViewControllerDelegate - @objc public weak var delegate: STPShippingAddressViewControllerDelegate? - - /// If you're pushing `STPShippingAddressViewController` onto an existing `UINavigationController`'s stack, you should use this method to dismiss it, since it may have pushed an additional shipping method view controller onto the navigation controller's stack. - /// - Parameter completion: The callback to run after the view controller is dismissed. You may specify nil for this parameter. - @objc(dismissWithCompletion:) - public func dismiss(withCompletion completion: STPVoidBlock?) { - if stp_isAtRootOfNavigationController() { - presentingViewController?.dismiss(animated: true, completion: completion ?? {}) - } else { - var previous = navigationController?.viewControllers.first - for viewController in navigationController?.viewControllers ?? [] { - if viewController == self { - break - } - previous = viewController - } - navigationController?.stp_pop( - to: previous, - animated: true, - completion: completion ?? {} - ) - } - } - - /// Use one of the initializers declared in this interface. - @available( - *, - unavailable, - message: "Use one of the initializers declared in this interface instead." - ) - @objc public required init( - theme: STPTheme? - ) { - let configuration = STPPaymentConfiguration.shared - addressViewModel = STPAddressViewModel( - requiredBillingFields: configuration.requiredBillingAddressFields, - availableCountries: configuration.availableCountries - ) - - super.init(theme: theme) - } - - /// Use one of the initializers declared in this interface. - @available( - *, - unavailable, - message: "Use one of the initializers declared in this interface instead." - ) - @objc public required init( - nibName nibNameOrNil: String?, - bundle nibBundleOrNil: Bundle? - ) { - let configuration = STPPaymentConfiguration.shared - addressViewModel = STPAddressViewModel( - requiredBillingFields: configuration.requiredBillingAddressFields, - availableCountries: configuration.availableCountries - ) - super.init(nibName: nibNameOrNil, bundle: nibBundleOrNil) - } - - /// Use one of the initializers declared in this interface. - @available( - *, - unavailable, - message: "Use one of the initializers declared in this interface instead." - ) - required init?( - coder aDecoder: NSCoder - ) { - let configuration = STPPaymentConfiguration.shared - addressViewModel = STPAddressViewModel( - requiredBillingFields: configuration.requiredBillingAddressFields, - availableCountries: configuration.availableCountries - ) - super.init(coder: aDecoder) - } - - private var configuration: STPPaymentConfiguration? - private var currency: String? - private var selectedShippingMethod: PKShippingMethod? - private weak var imageView: UIImageView? - private var nextItem: UIBarButtonItem? - - private var _loading = false - private var loading: Bool { - get { - _loading - } - set(loading) { - if loading == _loading { - return - } - _loading = loading - stp_navigationItemProxy?.setHidesBackButton(loading, animated: true) - stp_navigationItemProxy?.leftBarButtonItem?.isEnabled = !loading - activityIndicator?.animating = loading - if loading { - tableView?.endEditing(true) - var loadingItem: UIBarButtonItem? - if let activityIndicator = activityIndicator { - loadingItem = UIBarButtonItem(customView: activityIndicator) - } - stp_navigationItemProxy?.setRightBarButton(loadingItem, animated: true) - } else { - stp_navigationItemProxy?.setRightBarButton(nextItem, animated: true) - } - for cell in addressViewModel.addressCells { - cell.isUserInteractionEnabled = !loading - UIView.animate( - withDuration: 0.1, - animations: { - cell.alpha = loading ? 0.7 : 1.0 - } - ) - } - } - } - private var activityIndicator: STPPaymentActivityIndicatorView? - internal var addressViewModel: STPAddressViewModel - private var billingAddress: STPAddress? - private var hasUsedBillingAddress = false - private var addressHeaderView: STPSectionHeaderView? - - override func createAndSetupViews() { - super.createAndSetupViews() - - var nextItem: UIBarButtonItem? - switch configuration?.shippingType { - case .shipping: - nextItem = UIBarButtonItem( - title: STPLocalizedString("Next", "Button to move to the next text entry field"), - style: .done, - target: self, - action: #selector(next(_:)) - ) - case .delivery, .none, .some: - nextItem = UIBarButtonItem( - barButtonSystemItem: .done, - target: self, - action: #selector(next(_:)) - ) - } - self.nextItem = nextItem - stp_navigationItemProxy?.rightBarButtonItem = nextItem - stp_navigationItemProxy?.rightBarButtonItem?.isEnabled = false - stp_navigationItemProxy?.rightBarButtonItem?.accessibilityIdentifier = - "ShippingViewControllerNextButtonIdentifier" - - let imageView = UIImageView(image: STPLegacyImageLibrary.largeShippingImage()) - imageView.contentMode = .center - imageView.frame = CGRect( - x: 0, - y: 0, - width: view.bounds.size.width, - height: imageView.bounds.size.height + (57 * 2) - ) - self.imageView = imageView - tableView?.tableHeaderView = imageView - - activityIndicator = STPPaymentActivityIndicatorView( - frame: CGRect(x: 0, y: 0, width: 20.0, height: 20.0) - ) - - tableView?.dataSource = self - tableView?.delegate = self - tableView?.reloadData() - view.addGestureRecognizer( - UITapGestureRecognizer( - target: self, - action: #selector(NSMutableAttributedString.endEditing) - ) - ) - - let headerView = STPSectionHeaderView() - headerView.theme = theme - if let shippingType1 = configuration?.shippingType { - headerView.title = headerTitle(for: shippingType1) - } - headerView.button?.setTitle( - STPLocalizedString( - "Use Billing", - "Button to fill shipping address from billing address." - ), - for: .normal - ) - headerView.button?.addTarget( - self, - action: #selector(useBillingAddress(_:)), - for: .touchUpInside - ) - headerView.button?.accessibilityIdentifier = "ShippingAddressViewControllerUseBillingButton" - var buttonVisible = false - if let requiredFields = configuration?.requiredShippingAddressFields { - let needsAddress = - requiredFields.contains(.postalAddress) && !(addressViewModel.isValid) - buttonVisible = - needsAddress - && billingAddress?.containsContent(forShippingAddressFields: requiredFields) - ?? false - && !hasUsedBillingAddress - } - headerView.button?.alpha = buttonVisible ? 1 : 0 - headerView.setNeedsLayout() - addressHeaderView = headerView - - updateDoneButton() - } - - @objc func endEditing() { - view.endEditing(false) - } - - @objc override func updateAppearance() { - super.updateAppearance() - let navBarTheme = navigationController?.navigationBar.stp_theme ?? theme - nextItem?.stp_setTheme(navBarTheme) - - tableView?.allowsSelection = false - - imageView?.tintColor = theme.accentColor - activityIndicator?.tintColor = theme.accentColor - for cell in addressViewModel.addressCells { - cell.theme = theme - } - addressHeaderView?.theme = theme - } - - /// :nodoc: - @objc - public override func viewDidAppear(_ animated: Bool) { - super.viewDidAppear(animated) - stp_beginObservingKeyboardAndInsettingScrollView( - tableView, - onChange: nil - ) - firstEmptyField()?.becomeFirstResponder() - } - - func firstEmptyField() -> UIResponder? { - for cell in addressViewModel.addressCells { - if (cell.contents?.count ?? 0) == 0 { - return cell - } - } - return nil - } - - @objc override func handleCancelTapped(_ sender: Any?) { - delegate?.shippingAddressViewControllerDidCancel(self) - } - - @objc func next(_ sender: Any?) { - let address = addressViewModel.address - switch configuration?.shippingType { - case .shipping: - loading = true - delegate?.shippingAddressViewController(self, didEnter: address) { - status, - shippingValidationError, - shippingMethods, - selectedShippingMethod in - self.loading = false - if status == .valid { - if (shippingMethods?.count ?? 0) > 0 { - var nextViewController: STPShippingMethodsViewController? - if let shippingMethods = shippingMethods, - let selectedShippingMethod = selectedShippingMethod - { - nextViewController = STPShippingMethodsViewController( - shippingMethods: shippingMethods, - selectedShippingMethod: selectedShippingMethod, - currency: self.currency ?? "", - theme: self.theme - ) - } - nextViewController?.delegate = self - if let nextViewController = nextViewController { - self.navigationController?.pushViewController( - nextViewController, - animated: true - ) - } - } else { - self.delegate?.shippingAddressViewController( - self, - didFinishWith: address, - shippingMethod: nil - ) - } - } else { - self.handleShippingValidationError(shippingValidationError) - } - } - case .delivery, .none, .some: - delegate?.shippingAddressViewController( - self, - didFinishWith: address, - shippingMethod: nil - ) - } - } - - func updateDoneButton() { - stp_navigationItemProxy?.rightBarButtonItem?.isEnabled = addressViewModel.isValid - } - - func handleShippingValidationError(_ error: Error?) { - firstEmptyField()?.becomeFirstResponder() - var title = STPLocalizedString("Invalid Shipping Address", "Shipping form error message") - var message: String? - if let error = error { - title = error.localizedDescription - message = (error as NSError).localizedFailureReason - } - let alertController = UIAlertController( - title: title, - message: message, - preferredStyle: .alert - ) - alertController.addAction( - UIAlertAction( - title: String.Localized.ok, - style: .cancel, - handler: nil - ) - ) - present(alertController, animated: true) - } - - /// :nodoc: - @objc - public override func tableView( - _ tableView: UITableView, - heightForHeaderInSection section: Int - ) -> CGFloat { - let size = addressHeaderView?.sizeThatFits( - CGSize(width: view.bounds.size.width, height: CGFloat.greatestFiniteMagnitude) - ) - return size?.height ?? 0.0 - } - - @objc func useBillingAddress(_ sender: UIButton) { - guard let billingAddress = billingAddress else { - return - } - tableView?.beginUpdates() - addressViewModel.address = billingAddress - hasUsedBillingAddress = true - firstEmptyField()?.becomeFirstResponder() - UIView.animate( - withDuration: 0.2, - animations: { - self.addressHeaderView?.buttonHidden = true - } - ) - tableView?.endUpdates() - } - - func title(for type: STPShippingType) -> String { - if let shippingAddressFields = configuration?.requiredShippingAddressFields, - shippingAddressFields.contains(.postalAddress) - { - switch type { - case .shipping: - return STPLocalizedString("Shipping", "Title for shipping info form") - case .delivery: - return STPLocalizedString("Delivery", "Title for delivery info form") - } - } else { - return STPLocalizedString("Contact", "Title for contact info form") - } - } - - func headerTitle(for type: STPShippingType) -> String { - if let shippingAddressFields = configuration?.requiredShippingAddressFields, - shippingAddressFields.contains(.postalAddress) - { - switch type { - case .shipping: - return String.Localized.shipping_address - case .delivery: - return STPLocalizedString( - "Delivery Address", - "Title for delivery address entry section" - ) - } - } else { - return STPLocalizedString("Contact", "Title for contact info form") - } - } -} - -/// An `STPShippingAddressViewControllerDelegate` is notified when an `STPShippingAddressViewController` receives an address, completes with an address, or is cancelled. -@objc public protocol STPShippingAddressViewControllerDelegate: NSObjectProtocol { - /// Called when the user cancels entering a shipping address. You should dismiss (or pop) the view controller at this point. - /// - Parameter addressViewController: the view controller that has been cancelled - func shippingAddressViewControllerDidCancel( - _ addressViewController: STPShippingAddressViewController - ) - /// This is called when the user enters a shipping address and taps next. You - /// should validate the address and determine what shipping methods are available, - /// and call the `completion` block when finished. If an error occurrs, call - /// the `completion` block with the error. Otherwise, call the `completion` - /// block with a nil error and an array of available shipping methods. If you don't - /// need to collect a shipping method, you may pass an empty array or nil. - /// - Parameters: - /// - addressViewController: the view controller where the address was entered - /// - address: the address that was entered. - seealso: STPAddress - /// - completion: call this callback when you're done validating the address and determining available shipping methods. - - @objc(shippingAddressViewController:didEnterAddress:completion:) - func shippingAddressViewController( - _ addressViewController: STPShippingAddressViewController, - didEnter address: STPAddress, - completion: @escaping STPShippingMethodsCompletionBlock - ) - /// This is called when the user selects a shipping method. If no shipping methods are given, or if the shipping type doesn't require a shipping method, this will be called after the user has a shipping address and your validation has succeeded. After updating your app with the user's shipping info, you should dismiss (or pop) the view controller. Note that if `shippingMethod` is non-nil, there will be an additional shipping methods view controller on the navigation controller's stack. - /// - Parameters: - /// - addressViewController: the view controller where the address was entered - /// - address: the address that was entered. - seealso: STPAddress - /// - method: the shipping method that was selected. - @objc(shippingAddressViewController:didFinishWithAddress:shippingMethod:) - func shippingAddressViewController( - _ addressViewController: STPShippingAddressViewController, - didFinishWith address: STPAddress, - shippingMethod method: PKShippingMethod? - ) -} - -extension STPShippingAddressViewController: - STPAddressViewModelDelegate, UITableViewDelegate, UITableViewDataSource, - STPShippingMethodsViewControllerDelegate -{ - - // MARK: - UITableView - /// :nodoc: - @objc - public func numberOfSections(in tableView: UITableView) -> Int { - return 1 - } - - /// :nodoc: - @objc - public func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int { - return addressViewModel.addressCells.count - } - - /// :nodoc: - @objc - public func tableView( - _ tableView: UITableView, - cellForRowAt indexPath: IndexPath - ) -> UITableViewCell { - let cell = - addressViewModel.addressCells.stp_boundSafeObject(at: indexPath.row) - cell?.backgroundColor = theme.secondaryBackgroundColor - cell?.contentView.backgroundColor = UIColor.clear - return cell! - } - - /// :nodoc: - @objc - public func tableView( - _ tableView: UITableView, - willDisplay cell: UITableViewCell, - forRowAt indexPath: IndexPath - ) { - let topRow = indexPath.row == 0 - let bottomRow = tableView.numberOfRows(inSection: indexPath.section) - 1 == indexPath.row - cell.stp_setBorderColor(theme.tertiaryBackgroundColor) - cell.stp_setTopBorderHidden(!topRow) - cell.stp_setBottomBorderHidden(!bottomRow) - cell.stp_setFakeSeparatorColor(theme.quaternaryBackgroundColor) - cell.stp_setFakeSeparatorLeftInset(15.0) - } - - /// :nodoc: - @objc - public func tableView( - _ tableView: UITableView, - heightForFooterInSection section: Int - ) - -> CGFloat - { - return 0.01 - } - - /// :nodoc: - @objc - public func tableView( - _ tableView: UITableView, - viewForFooterInSection section: Int - ) - -> UIView? - { - return UIView() - } - - /// :nodoc: - @objc - public func tableView( - _ tableView: UITableView, - viewForHeaderInSection section: Int - ) - -> UIView? - { - return addressHeaderView - } - - // MARK: - STPShippingMethodsViewControllerDelegate - func shippingMethodsViewController( - _ methodsViewController: STPShippingMethodsViewController, - didFinishWith method: PKShippingMethod - ) { - delegate?.shippingAddressViewController( - self, - didFinishWith: addressViewModel.address, - shippingMethod: method - ) - } - - // MARK: - STPAddressViewModelDelegate - func addressViewModel(_ addressViewModel: STPAddressViewModel, addedCellAt index: Int) { - let indexPath = IndexPath(row: index, section: 0) - tableView?.insertRows(at: [indexPath], with: .automatic) - } - - func addressViewModel(_ addressViewModel: STPAddressViewModel, removedCellAt index: Int) { - let indexPath = IndexPath(row: index, section: 0) - tableView?.deleteRows(at: [indexPath], with: .automatic) - } - - func addressViewModelDidChange(_ addressViewModel: STPAddressViewModel) { - updateDoneButton() - } - - func addressViewModelWillUpdate(_ addressViewModel: STPAddressViewModel) { - tableView?.beginUpdates() - } - - func addressViewModelDidUpdate(_ addressViewModel: STPAddressViewModel) { - tableView?.endUpdates() - } -} - -/// :nodoc: -@_spi(STP) extension STPShippingAddressViewController: STPAnalyticsProtocol { - @_spi(STP) public static var stp_analyticsIdentifier = "STPShippingAddressViewController" -} diff --git a/Stripe/StripeiOS/Source/STPShippingMethodsViewController.swift b/Stripe/StripeiOS/Source/STPShippingMethodsViewController.swift deleted file mode 100644 index 8d915855419..00000000000 --- a/Stripe/StripeiOS/Source/STPShippingMethodsViewController.swift +++ /dev/null @@ -1,211 +0,0 @@ -// -// STPShippingMethodsViewController.swift -// StripeiOS -// -// Created by Ben Guo on 8/29/16. -// Copyright © 2016 Stripe, Inc. All rights reserved. -// - -import PassKit -@_spi(STP) import StripeCore -import UIKit - -class STPShippingMethodsViewController: STPCoreTableViewController, UITableViewDataSource, - UITableViewDelegate -{ - init( - shippingMethods methods: [PKShippingMethod], - selectedShippingMethod selectedMethod: PKShippingMethod, - currency: String, - theme: STPTheme - ) { - super.init(theme: theme) - shippingMethods = methods - if (methods.firstIndex(of: selectedMethod) ?? NSNotFound) != NSNotFound { - selectedShippingMethod = selectedMethod - } else { - selectedShippingMethod = methods.stp_boundSafeObject(at: 0) - } - - self.currency = currency - title = STPLocalizedString("Shipping", "Title for shipping info form") - } - - weak var delegate: STPShippingMethodsViewControllerDelegate? - private var shippingMethods: [PKShippingMethod]? - private var selectedShippingMethod: PKShippingMethod? - private var currency: String? - private weak var imageView: UIImageView? - private var doneItem: UIBarButtonItem? - private var headerView: STPSectionHeaderView? - - override func createAndSetupViews() { - super.createAndSetupViews() - - tableView?.register( - STPShippingMethodTableViewCell.self, - forCellReuseIdentifier: STPShippingMethodCellReuseIdentifier - ) - - let doneItem = UIBarButtonItem( - barButtonSystemItem: .done, - target: self, - action: #selector(done(_:)) - ) - self.doneItem = doneItem - stp_navigationItemProxy?.rightBarButtonItem = doneItem - stp_navigationItemProxy?.rightBarButtonItem?.accessibilityIdentifier = - "ShippingMethodsViewControllerDoneButtonIdentifier" - - let imageView = UIImageView(image: STPLegacyImageLibrary.largeShippingImage()) - imageView.contentMode = .center - imageView.frame = CGRect( - x: 0, - y: 0, - width: view.bounds.size.width, - height: imageView.bounds.size.height + (57 * 2) - ) - self.imageView = imageView - - tableView?.tableHeaderView = imageView - tableView?.dataSource = self - tableView?.delegate = self - tableView?.reloadData() - - let headerView = STPSectionHeaderView() - headerView.theme = theme - headerView.buttonHidden = true - headerView.title = STPLocalizedString("Shipping Method", "Label for shipping method form") - headerView.setNeedsLayout() - self.headerView = headerView - } - - @objc override func updateAppearance() { - super.updateAppearance() - - let navBarTheme = navigationController?.navigationBar.stp_theme ?? theme - doneItem?.stp_setTheme(navBarTheme) - - imageView?.tintColor = theme.accentColor - for cell in tableView?.visibleCells ?? [] { - let shippingCell = cell as? STPShippingMethodTableViewCell - shippingCell?.theme = theme - } - } - - @objc func done(_ sender: Any?) { - if let selectedShippingMethod = selectedShippingMethod { - delegate?.shippingMethodsViewController(self, didFinishWith: selectedShippingMethod) - } - } - - override func useSystemBackButton() -> Bool { - return true - } - - // MARK: - UITableView - func numberOfSections(in tableView: UITableView) -> Int { - return 1 - } - - func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int { - return shippingMethods?.count ?? 0 - } - - func tableView( - _ tableView: UITableView, - cellForRowAt indexPath: IndexPath - ) -> UITableViewCell { - let cell = - tableView.dequeueReusableCell( - withIdentifier: STPShippingMethodCellReuseIdentifier, - for: indexPath - ) - as? STPShippingMethodTableViewCell - let method = - shippingMethods?.stp_boundSafeObject(at: indexPath.row) - cell?.theme = theme - if let method = method { - cell?.setShippingMethod(method, currency: currency ?? "") - } - cell?.isSelected = method?.identifier == selectedShippingMethod?.identifier - return cell! - } - - func tableView( - _ tableView: UITableView, - willDisplay cell: UITableViewCell, - forRowAt indexPath: IndexPath - ) { - let topRow = indexPath.row == 0 - let bottomRow = - self.tableView(tableView, numberOfRowsInSection: indexPath.section) - 1 == indexPath.row - cell.stp_setBorderColor(theme.tertiaryBackgroundColor) - cell.stp_setTopBorderHidden(!topRow) - cell.stp_setBottomBorderHidden(!bottomRow) - cell.stp_setFakeSeparatorColor(theme.quaternaryBackgroundColor) - cell.stp_setFakeSeparatorLeftInset(15.0) - } - - func tableView(_ tableView: UITableView, heightForRowAt indexPath: IndexPath) -> CGFloat { - return 57 - } - - func tableView(_ tableView: UITableView, heightForFooterInSection section: Int) -> CGFloat { - return 27.0 - } - - override func tableView( - _ tableView: UITableView, - heightForHeaderInSection section: Int - ) - -> CGFloat - { - let size = headerView?.sizeThatFits( - CGSize(width: view.bounds.size.width, height: CGFloat.greatestFiniteMagnitude) - ) - return size?.height ?? 0.0 - } - - func tableView(_ tableView: UITableView, viewForHeaderInSection section: Int) -> UIView? { - return headerView - } - - func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) { - tableView.deselectRow(at: indexPath, animated: true) - selectedShippingMethod = - shippingMethods?.stp_boundSafeObject(at: indexPath.row) - tableView.reloadSections( - NSIndexSet(index: indexPath.section) as IndexSet, - with: .fade - ) - } - - required init?( - coder aDecoder: NSCoder - ) { - super.init(coder: aDecoder) - } - - required init( - nibName nibNameOrNil: String?, - bundle nibBundleOrNil: Bundle? - ) { - fatalError("init(nibName:bundle:) has not been implemented") - } - - required init( - theme: STPTheme? - ) { - fatalError("init(theme:) has not been implemented") - } -} - -@objc protocol STPShippingMethodsViewControllerDelegate: NSObjectProtocol { - func shippingMethodsViewController( - _ methodsViewController: STPShippingMethodsViewController, - didFinishWith method: PKShippingMethod - ) -} - -private let STPShippingMethodCellReuseIdentifier = "STPShippingMethodCellReuseIdentifier" diff --git a/Stripe/StripeiOSTests/STPAddCardViewControllerLocalizationSnapshotTests.swift b/Stripe/StripeiOSTests/STPAddCardViewControllerLocalizationSnapshotTests.swift deleted file mode 100644 index 9ebaf74d8a0..00000000000 --- a/Stripe/StripeiOSTests/STPAddCardViewControllerLocalizationSnapshotTests.swift +++ /dev/null @@ -1,101 +0,0 @@ -// -// STPAddCardViewControllerLocalizationSnapshotTests.swift -// StripeiOS Tests -// -// Created by Brian Dorfman on 10/17/16. -// Copyright © 2016 Stripe, Inc. All rights reserved. -// - -import iOSSnapshotTestCase -import StripeCoreTestUtils - -@testable@_spi(STP) import Stripe -@testable@_spi(STP) import StripeCore -@testable@_spi(STP) import StripePayments -@testable@_spi(STP) import StripePaymentSheet -@testable@_spi(STP) import StripePaymentsUI - -class STPAddCardViewControllerLocalizationSnapshotTests: STPSnapshotTestCase { - func performSnapshotTest(forLanguage language: String?, delivery: Bool) { - let config = STPPaymentConfiguration() - config.companyName = "Test Company" - config.requiredBillingAddressFields = .full - config.shippingType = delivery ? .delivery : .shipping - config.cardScanningEnabled = true - STPLocalizationUtils.overrideLanguage(to: language) - - let addCardVC = STPAddCardViewController( - configuration: config, - theme: STPTheme.defaultTheme - ) - addCardVC.shippingAddress = STPAddress() - addCardVC.shippingAddress?.line1 = "1" // trigger "use shipping address" button - - let viewToTest = stp_preparedAndSizedViewForSnapshotTest(from: addCardVC)! - - if delivery { - addCardVC.addressViewModel.addressFieldTableViewCountryCode = "INVALID" - STPSnapshotVerifyView(viewToTest, identifier: "delivery") - } else { - // This method rejects nil or empty country codes to stop strange looking behavior - // when scrolling to the top "unset" position in the picker, so put in - // an invalid country code instead to test seeing the "Country" placeholder - addCardVC.addressViewModel.addressFieldTableViewCountryCode = "INVALID" - STPSnapshotVerifyView(viewToTest, identifier: "no_country") - - addCardVC.addressViewModel.addressFieldTableViewCountryCode = "US" - STPSnapshotVerifyView(viewToTest, identifier: "US") - - addCardVC.addressViewModel.addressFieldTableViewCountryCode = "GB" - STPSnapshotVerifyView(viewToTest, identifier: "GB") - - addCardVC.addressViewModel.addressFieldTableViewCountryCode = "CA" - STPSnapshotVerifyView(viewToTest, identifier: "CA") - - addCardVC.addressViewModel.addressFieldTableViewCountryCode = "MX" - STPSnapshotVerifyView(viewToTest, identifier: "MX") - } - - STPLocalizationUtils.overrideLanguage(to: nil) - } - - func testGerman() { - performSnapshotTest(forLanguage: "de", delivery: false) - performSnapshotTest(forLanguage: "de", delivery: true) - } - - func testEnglish() { - performSnapshotTest(forLanguage: "en", delivery: false) - performSnapshotTest(forLanguage: "en", delivery: true) - } - - func testSpanish() { - performSnapshotTest(forLanguage: "es", delivery: false) - performSnapshotTest(forLanguage: "es", delivery: true) - } - - func testFrench() { - performSnapshotTest(forLanguage: "fr", delivery: false) - performSnapshotTest(forLanguage: "fr", delivery: true) - } - - func testItalian() { - performSnapshotTest(forLanguage: "it", delivery: false) - performSnapshotTest(forLanguage: "it", delivery: true) - } - - func testJapanese() { - performSnapshotTest(forLanguage: "ja", delivery: false) - performSnapshotTest(forLanguage: "ja", delivery: true) - } - - func testDutch() { - performSnapshotTest(forLanguage: "nl", delivery: false) - performSnapshotTest(forLanguage: "nl", delivery: true) - } - - func testChinese() { - performSnapshotTest(forLanguage: "zh-Hans", delivery: false) - performSnapshotTest(forLanguage: "zh-Hans", delivery: true) - } -} diff --git a/Stripe/StripeiOSTests/STPAddCardViewControllerTest.swift b/Stripe/StripeiOSTests/STPAddCardViewControllerTest.swift deleted file mode 100644 index 16b50807c90..00000000000 --- a/Stripe/StripeiOSTests/STPAddCardViewControllerTest.swift +++ /dev/null @@ -1,290 +0,0 @@ -// -// STPAddCardViewControllerTest.swift -// StripeiOS Tests -// -// Created by Ben Guo on 7/5/16. -// Copyright © 2016 Stripe, Inc. All rights reserved. -// - -import OHHTTPStubs -import OHHTTPStubsSwift -import StripeCoreTestUtils - -@testable@_spi(STP) import Stripe -@testable@_spi(STP) import StripeCore -@testable@_spi(STP) import StripePayments -@testable@_spi(STP) import StripePaymentSheet -@testable@_spi(STP) import StripePaymentsUI - -class MockDelegate: NSObject, STPAddCardViewControllerDelegate { - func addCardViewControllerDidCancel(_ addCardViewController: STPAddCardViewController) { - - } - - var addCardViewControllerDidCreatePaymentMethodBlock: - (STPAddCardViewController, STPPaymentMethod, STPErrorBlock) -> Void = { _, _, _ in } - func addCardViewController( - _ addCardViewController: STPAddCardViewController, - didCreatePaymentMethod paymentMethod: STPPaymentMethod, - completion: @escaping STPErrorBlock - ) { - addCardViewControllerDidCreatePaymentMethodBlock( - addCardViewController, - paymentMethod, - completion - ) - } -} - -class STPAddCardViewControllerTest: APIStubbedTestCase { - - func paymentMethodAPIFilter( - expectedCardParams: STPPaymentMethodCardParams, - urlRequest: URLRequest - ) -> Bool { - if urlRequest.url?.absoluteString.contains("payment_methods") ?? false { - let cardNumber = urlRequest.queryItems?.first(where: { item in - item.name == "card[number]" - }) - XCTAssertEqual(cardNumber!.value, expectedCardParams.number) - return true - } - return false - } - - func buildAddCardViewController() -> STPAddCardViewController? { - let config = STPPaymentConfiguration() - let theme = STPTheme.defaultTheme - let vc = STPAddCardViewController( - configuration: config, - theme: theme - ) - XCTAssertNotNil(vc.view) - return vc - } - - func testPrefilledBillingAddress_removeAddress() { - let config = STPPaymentConfiguration() - config.requiredBillingAddressFields = .postalCode - let sut = STPAddCardViewController( - configuration: config, - theme: STPTheme.defaultTheme - ) - let address = STPAddress() - address.name = "John Smith Doe" - address.phone = "8885551212" - address.email = "foo@example.com" - address.line1 = "55 John St" - address.city = "Harare" - address.postalCode = "10002" - // Zimbabwe does not require zip codes, while the default locale for tests (US) does - address.country = "ZW" - // Sanity checks - XCTAssertFalse(STPPostalCodeValidator.postalCodeIsRequired(forCountryCode: "ZW")) - XCTAssertTrue(STPPostalCodeValidator.postalCodeIsRequired(forCountryCode: "US")) - - let prefilledInfo = STPUserInformation() - prefilledInfo.billingAddress = address - sut.prefilledInformation = prefilledInfo - - XCTAssertNoThrow(sut.loadView()) - XCTAssertNoThrow(sut.viewDidLoad()) - } - - func testPrefilledBillingAddress_viewDidLoadHappensBeforeSettingAddress() { - let config = STPPaymentConfiguration() - config.requiredBillingAddressFields = .full - let sut = STPAddCardViewController( - configuration: config, - theme: STPTheme.defaultTheme - ) - XCTAssertNoThrow(sut.loadView()) - XCTAssertNoThrow(sut.viewDidLoad()) - - let address = STPAddress() - address.name = "John Smith Doe" - address.line1 = "55 John St" - address.city = "Harare" - address.postalCode = "10002" - - let prefilledInfo = STPUserInformation() - prefilledInfo.billingAddress = address - sut.prefilledInformation = prefilledInfo - - let nameCell = sut.addressViewModel.addressCells.first { $0.type == .name }! - XCTAssertEqual( nameCell.contents, "John Smith Doe") - - let line1Cell = sut.addressViewModel.addressCells.first { $0.type == .line1 }! - XCTAssertEqual( line1Cell.contents, "55 John St") - - let cityCell = sut.addressViewModel.addressCells.first { $0.type == .city }! - XCTAssertEqual( cityCell.contents, "Harare") - - let zipCell = sut.addressViewModel.addressCells.first { $0.type == .zip }! - XCTAssertEqual( zipCell.contents, "10002") - } - - func testPrefilledBillingAddress_addAddress() { - // Zimbabwe does not require zip codes, while the default locale for tests (US) does - NSLocale.stp_withLocale(as: NSLocale(localeIdentifier: "en_ZW")) { - // Sanity checks - XCTAssertFalse(STPPostalCodeValidator.postalCodeIsRequired(forCountryCode: "ZW")) - XCTAssertTrue(STPPostalCodeValidator.postalCodeIsRequired(forCountryCode: "US")) - let config = STPPaymentConfiguration() - config.requiredBillingAddressFields = .postalCode - let sut = STPAddCardViewController( - configuration: config, - theme: STPTheme.defaultTheme - ) - let address = STPAddress() - address.name = "John Smith Doe" - address.phone = "8885551212" - address.email = "foo@example.com" - address.line1 = "55 John St" - address.city = "New York" - address.state = "NY" - address.postalCode = "10002" - address.country = "US" - - let prefilledInfo = STPUserInformation() - prefilledInfo.billingAddress = address - sut.prefilledInformation = prefilledInfo - - XCTAssertNoThrow(sut.loadView()) - XCTAssertNoThrow(sut.viewDidLoad()) - } - } - - func testNextWithCreatePaymentMethodError() { - let sut = buildAddCardViewController()! - let expectedCardParams = STPFixtures.paymentMethodCardParams() - sut.paymentCell?.paymentField!.perform(NSSelectorFromString("setCardParams:"), with: expectedCardParams) - - let exp = expectation(description: "createPaymentMethodWithCard network request") - stub { urlRequest in - return self.paymentMethodAPIFilter( - expectedCardParams: expectedCardParams, - urlRequest: urlRequest - ) - } response: { _ in - XCTAssertTrue(sut.loading) - let paymentMethod = ["error": "intentionally_invalid"] - defer { - exp.fulfill() - } - return HTTPStubsResponse(jsonObject: paymentMethod, statusCode: 200, headers: nil) - } - sut.apiClient = stubbedAPIClient() - // tap next button - let nextButton = sut.navigationItem.rightBarButtonItem - _ = nextButton?.target?.perform(nextButton?.action, with: nextButton) - - waitForExpectations(timeout: 2, handler: nil) - - // It takes a few more spins on the runloop before we get a response from - // the HTTP stubs, so we'll wait 0.5 seconds before checking the loading indicator. - let loadExp = expectation(description: "loading has stopped") - DispatchQueue.main.asyncAfter(deadline: .now() + 0.5) { - XCTAssertFalse(sut.loading) - loadExp.fulfill() - } - waitForExpectations(timeout: 1, handler: nil) - } - - func testNextWithCreatePaymentMethodSuccessAndDidCreatePaymentMethodError() { - let sut = buildAddCardViewController()! - let createPaymentMethodExp = expectation(description: "createPaymentMethodWithCard") - - let expectedCardParams = STPFixtures.paymentMethodCardParams() - let expectedPaymentMethod = STPFixtures.paymentMethod() - let expectedPaymentMethodData = STPFixtures.paymentMethodJSON() - - stub { urlRequest in - return self.paymentMethodAPIFilter( - expectedCardParams: expectedCardParams, - urlRequest: urlRequest - ) - } response: { _ in - XCTAssertTrue(sut.loading) - defer { - createPaymentMethodExp.fulfill() - } - return HTTPStubsResponse( - jsonObject: expectedPaymentMethodData, - statusCode: 200, - headers: nil - ) - } - - let mockDelegate = MockDelegate() - sut.apiClient = stubbedAPIClient() - sut.delegate = mockDelegate - sut.paymentCell?.paymentField!.perform(NSSelectorFromString("setCardParams:"), with: expectedCardParams) - - let didCreatePaymentMethodExp = expectation(description: "didCreatePaymentMethod") - - mockDelegate.addCardViewControllerDidCreatePaymentMethodBlock = { - (_, paymentMethod, completion) in - XCTAssertTrue(sut.loading) - let error = NSError.stp_genericFailedToParseResponseError() - XCTAssertEqual(paymentMethod.stripeId, expectedPaymentMethod.stripeId) - completion(error) - XCTAssertFalse(sut.loading) - didCreatePaymentMethodExp.fulfill() - } - - // tap next button - let nextButton = sut.navigationItem.rightBarButtonItem - _ = nextButton?.target?.perform(nextButton?.action, with: nextButton) - - waitForExpectations(timeout: 2, handler: nil) - } - - func testNextWithCreateTokenSuccessAndDidCreateTokenSuccess() { - let sut = buildAddCardViewController()! - - let createPaymentMethodExp = expectation(description: "createPaymentMethodWithCard") - - let expectedCardParams = STPFixtures.paymentMethodCardParams() - let expectedPaymentMethod = STPFixtures.paymentMethod() - let expectedPaymentMethodData = STPFixtures.paymentMethodJSON() - - stub { urlRequest in - return self.paymentMethodAPIFilter( - expectedCardParams: expectedCardParams, - urlRequest: urlRequest - ) - } response: { _ in - XCTAssertTrue(sut.loading) - defer { - createPaymentMethodExp.fulfill() - } - return HTTPStubsResponse( - jsonObject: expectedPaymentMethodData, - statusCode: 200, - headers: nil - ) - } - - let mockDelegate = MockDelegate() - sut.apiClient = stubbedAPIClient() - sut.delegate = mockDelegate - sut.paymentCell?.paymentField!.perform(NSSelectorFromString("setCardParams:"), with: expectedCardParams) - - let didCreatePaymentMethodExp = expectation(description: "didCreatePaymentMethod") - mockDelegate.addCardViewControllerDidCreatePaymentMethodBlock = { - (_, paymentMethod, completion) in - XCTAssertTrue(sut.loading) - XCTAssertEqual(paymentMethod.stripeId, expectedPaymentMethod.stripeId) - completion(nil) - XCTAssertFalse(sut.loading) - didCreatePaymentMethodExp.fulfill() - } - - // tap next button - let nextButton = sut.navigationItem.rightBarButtonItem - _ = nextButton?.target?.perform(nextButton?.action, with: nextButton) - - waitForExpectations(timeout: 2, handler: nil) - } -} diff --git a/Stripe/StripeiOSTests/STPAnalyticsClientPaymentsTest.swift b/Stripe/StripeiOSTests/STPAnalyticsClientPaymentsTest.swift index 5ab6e8b508d..1a4c14101b7 100644 --- a/Stripe/StripeiOSTests/STPAnalyticsClientPaymentsTest.swift +++ b/Stripe/StripeiOSTests/STPAnalyticsClientPaymentsTest.swift @@ -125,80 +125,17 @@ class STPAnalyticsClientPaymentsTest: XCTestCase { ) } - func testPaymentContextAddsUsage() { - let keyManager = STPEphemeralKeyManager( - keyProvider: MockKeyProvider(), - apiVersion: "1", - performsEagerFetching: false - ) - let apiClient = STPAPIClient() - let customerContext = STPCustomerContext.init(keyManager: keyManager, apiClient: apiClient) - let paymentContext = STPPaymentContext(customerContext: customerContext) - XCTAssertTrue(STPAnalyticsClient.sharedClient.productUsage.contains("STPCustomerContext")) - XCTAssertEqual(paymentContext.analyticsLogger.product, "STPPaymentContext") - } - func testApplePayContextAddsUsage() { _ = STPApplePayContext(paymentRequest: STPFixtures.applePayRequest(), delegate: nil) XCTAssertTrue(STPAnalyticsClient.sharedClient.productUsage.contains("STPApplePayContext")) } - func testCustomerContextAddsUsage() { - let keyManager = STPEphemeralKeyManager( - keyProvider: MockKeyProvider(), - apiVersion: "1", - performsEagerFetching: false - ) - let apiClient = STPAPIClient() - _ = STPCustomerContext(keyManager: keyManager, apiClient: apiClient) - XCTAssertTrue(STPAnalyticsClient.sharedClient.productUsage.contains("STPCustomerContext")) - } - - func testAddCardVCAddsUsage() { - let addCardVC = STPAddCardViewController() - XCTAssertTrue( - STPAnalyticsClient.sharedClient.productUsage.contains("STPAddCardViewController") - ) - XCTAssertEqual(addCardVC.analyticsLogger.product, "STPAddCardViewController") - } - - func testPaymentOptionsVCAddsUsage() { - let customerContext = Testing_StaticCustomerContext.init( - customer: STPFixtures.customerWithCardTokenAndSourceSources(), - paymentMethods: [] - ) - let delegate = MockSTPPaymentOptionsViewControllerDelegate() - let paymentOptionsVC = STPPaymentOptionsViewController(configuration: .shared, theme: .defaultTheme, customerContext: customerContext, delegate: delegate) - XCTAssertTrue( - STPAnalyticsClient.sharedClient.productUsage.contains("STPPaymentOptionsViewController") - ) - XCTAssertEqual(paymentOptionsVC.analyticsLogger.product, "STPPaymentOptionsViewController") - } - func testBankSelectionVCAddsUsage() { _ = STPBankSelectionViewController() XCTAssertTrue( STPAnalyticsClient.sharedClient.productUsage.contains("STPBankSelectionViewController") ) } - - func testShippingVCAddsUsage() { - let config = STPPaymentConfiguration() - config.requiredShippingAddressFields = [STPContactField.postalAddress] - _ = STPShippingAddressViewController( - configuration: config, - theme: .defaultTheme, - currency: nil, - shippingAddress: nil, - selectedShippingMethod: nil, - prefilledInformation: nil - ) - XCTAssertTrue( - STPAnalyticsClient.sharedClient.productUsage.contains( - "STPShippingAddressViewController" - ) - ) - } } // MARK: - Helpers diff --git a/Stripe/StripeiOSTests/STPCustomerContextTest.swift b/Stripe/StripeiOSTests/STPCustomerContextTest.swift deleted file mode 100644 index f622ac96ce8..00000000000 --- a/Stripe/StripeiOSTests/STPCustomerContextTest.swift +++ /dev/null @@ -1,671 +0,0 @@ -// -// STPCustomerContextTest.swift -// StripeiOS Tests -// -// Created by David Estes on 9/20/21. -// Copyright © 2021 Stripe, Inc. All rights reserved. -// - -import Foundation -import OHHTTPStubs -import OHHTTPStubsSwift -import StripeCoreTestUtils - -@testable@_spi(STP) import Stripe -@testable@_spi(STP) import StripeCore -@testable@_spi(STP) import StripePayments - -class MockEphemeralKeyManager: STPEphemeralKeyManagerProtocol { - var ephemeralKey: STPEphemeralKey? - var error: Error? - - init( - key: STPEphemeralKey?, - error: Error? - ) { - self.ephemeralKey = key - self.error = error - } - - func getOrCreateKey(_ completion: @escaping STPEphemeralKeyCompletionBlock) { - completion(ephemeralKey, error) - } -} - -class STPCustomerContextTests: APIStubbedTestCase { - func stubRetrieveCustomers( - key: STPEphemeralKey, - returningCustomerJSON: [AnyHashable: Any], - expectedCount: Int, - apiClient: STPAPIClient - ) { - let exp = expectation(description: "retrieveCustomer") - exp.expectedFulfillmentCount = expectedCount - - stub { urlRequest in - return urlRequest.url?.absoluteString.contains("/customers") ?? false - && urlRequest.httpMethod == "GET" - } response: { _ in - DispatchQueue.main.async { - // Fulfill after response is sent - exp.fulfill() - } - return HTTPStubsResponse( - jsonObject: returningCustomerJSON, - statusCode: 200, - headers: nil - ) - } - } - - func stubListPaymentMethods( - key: STPEphemeralKey, - paymentMethodJSONs: [[AnyHashable: Any]], - expectedCount: Int, - apiClient: STPAPIClient - ) { - let exp = expectation(description: "listPaymentMethod") - exp.expectedFulfillmentCount = expectedCount - stub { urlRequest in - if urlRequest.url?.absoluteString.contains("/payment_methods") ?? false - && urlRequest.httpMethod == "GET" - { - // Check to make sure we pass the ephemeral key correctly - let keyFromHeader = urlRequest.allHTTPHeaderFields!["Authorization"]? - .replacingOccurrences(of: "Bearer ", with: "") - XCTAssertEqual(keyFromHeader, key.secret) - return true - } - return false - } response: { _ in - let paymentMethodsJSON = """ - { - "object": "list", - "url": "/v1/payment_methods", - "has_more": false, - "data": [ - ] - } - """ - var pmList = - try! JSONSerialization.jsonObject( - with: paymentMethodsJSON.data(using: .utf8)!, - options: [] - ) as! [AnyHashable: Any] - pmList["data"] = paymentMethodJSONs - DispatchQueue.main.async { - // Fulfill after response is sent - exp.fulfill() - } - return HTTPStubsResponse(jsonObject: pmList, statusCode: 200, headers: nil) - } - } - - func testGetOrCreateKeyErrorForwardedToRetrieveCustomer() { - let exp = expectation(description: "retrieveCustomer") - let expectedError = NSError(domain: "test", code: 123, userInfo: nil) - let apiClient = stubbedAPIClient() - stub { urlRequest in - return urlRequest.url?.absoluteString.contains("/customers") ?? false - } response: { _ in - XCTFail("Retrieve customer should not be called") - return HTTPStubsResponse(error: NSError(domain: "test", code: 100, userInfo: nil)) - } - let ekm = MockEphemeralKeyManager(key: nil, error: expectedError) - let sut = STPCustomerContext(keyManager: ekm, apiClient: apiClient) - sut.retrieveCustomer { customer, error in - XCTAssertNil(customer) - XCTAssertEqual((error as NSError?)?.domain, expectedError.domain) - exp.fulfill() - } - waitForExpectations(timeout: 2, handler: nil) - } - - func testInitRetrievesResourceKeyAndCustomerAndPaymentMethods() { - let customerKey = STPFixtures.ephemeralKey() - let expectedCustomerJSON = STPFixtures.customerWithSingleCardTokenSourceJSON() - let apiClient = stubbedAPIClient() - stubRetrieveCustomers( - key: customerKey, - returningCustomerJSON: expectedCustomerJSON, - expectedCount: 1, - apiClient: apiClient - ) - stubListPaymentMethods( - key: customerKey, - paymentMethodJSONs: [], - expectedCount: 1, - apiClient: apiClient - ) - - let ekm = MockEphemeralKeyManager(key: customerKey, error: nil) - let sut = STPCustomerContext(keyManager: ekm, apiClient: apiClient) - XCTAssertNotNil(sut) - waitForExpectations(timeout: 2, handler: nil) - } - - func testRetrieveCustomerUsesCachedCustomerIfNotExpired() { - let customerKey = STPFixtures.ephemeralKey() - let expectedCustomer = STPFixtures.customerWithSingleCardTokenSource() - let expectedCustomerJSON = STPFixtures.customerWithSingleCardTokenSourceJSON() - let apiClient = stubbedAPIClient() - - // apiClient.retrieveCustomer should be called once, when the context is initialized. - // When sut.retrieveCustomer is called below, the cached customer will be used. - stubRetrieveCustomers( - key: customerKey, - returningCustomerJSON: expectedCustomerJSON, - expectedCount: 1, - apiClient: apiClient - ) - stubListPaymentMethods( - key: customerKey, - paymentMethodJSONs: [], - expectedCount: 1, - apiClient: apiClient - ) - let ekm = MockEphemeralKeyManager(key: customerKey, error: nil) - let sut = STPCustomerContext(keyManager: ekm, apiClient: apiClient) - // Give the mocked API request a little time to complete and cache the customer, then check cache - waitForExpectations(timeout: 2, handler: nil) - let exp2 = expectation(description: "retrieveCustomer again") - DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + ohhttpDelay) { - sut.retrieveCustomer { customer, _ in - XCTAssertEqual(customer!.stripeID, expectedCustomer.stripeID) - exp2.fulfill() - } - } - waitForExpectations(timeout: 2, handler: nil) - } - - func testRetrieveCustomerDoesNotUseCachedCustomerIfExpired() { - let customerKey = STPFixtures.ephemeralKey() - let expectedCustomer = STPFixtures.customerWithSingleCardTokenSource() - let expectedCustomerJSON = STPFixtures.customerWithSingleCardTokenSourceJSON() - let apiClient = stubbedAPIClient() - - // apiClient.retrieveCustomer should be called twice: - // - when the context is initialized, - // - when sut.retrieveCustomer is called below, as the cached customer has expired. - stubRetrieveCustomers( - key: customerKey, - returningCustomerJSON: expectedCustomerJSON, - expectedCount: 2, - apiClient: apiClient - ) - stubListPaymentMethods( - key: customerKey, - paymentMethodJSONs: [], - expectedCount: 1, - apiClient: apiClient - ) - - let ekm = MockEphemeralKeyManager(key: customerKey, error: nil) - let sut = STPCustomerContext(keyManager: ekm, apiClient: apiClient) - // Give the mocked API request a little time to complete and cache the customer, then reset and check cache - let exp2 = expectation(description: "retrieveCustomer again") - DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + ohhttpDelay) { - sut.customerRetrievedDate = Date(timeIntervalSinceNow: -70) - sut.retrieveCustomer { customer, _ in - XCTAssertEqual(customer!.stripeID, expectedCustomer.stripeID) - exp2.fulfill() - } - } - waitForExpectations(timeout: 2, handler: nil) - } - - func testRetrieveCustomerDoesNotUseCachedCustomerAfterClearingCache() { - let customerKey = STPFixtures.ephemeralKey() - let expectedCustomer = STPFixtures.customerWithSingleCardTokenSource() - let expectedCustomerJSON = STPFixtures.customerWithSingleCardTokenSourceJSON() - let apiClient = stubbedAPIClient() - - // apiClient.retrieveCustomer should be called twice: - // - when the context is initialized, - // - when sut.retrieveCustomer is called below, as the cached customer has been cleared. - stubRetrieveCustomers( - key: customerKey, - returningCustomerJSON: expectedCustomerJSON, - expectedCount: 2, - apiClient: apiClient - ) - stubListPaymentMethods( - key: customerKey, - paymentMethodJSONs: [], - expectedCount: 1, - apiClient: apiClient - ) - - let ekm = MockEphemeralKeyManager(key: customerKey, error: nil) - let sut = STPCustomerContext(keyManager: ekm, apiClient: apiClient) - // Give the mocked API request a little time to complete and cache the customer, then reset and check cache - let exp2 = expectation(description: "retrieveCustomer again") - DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + ohhttpDelay) { - sut.clearCache() - sut.retrieveCustomer { customer, _ in - XCTAssertEqual(customer!.stripeID, expectedCustomer.stripeID) - exp2.fulfill() - } - } - waitForExpectations(timeout: 2, handler: nil) - } - - func testRetrievePaymentMethodsUsesCacheIfNotExpired() { - let customerKey = STPFixtures.ephemeralKey() - let expectedCustomerJSON = STPFixtures.customerWithSingleCardTokenSourceJSON() - let expectedPaymentMethods = [STPFixtures.paymentMethod()] - let expectedPaymentMethodsJSON = [STPFixtures.paymentMethodJSON()] - let apiClient = stubbedAPIClient() - - // apiClient.listPaymentMethods should be called once, when the context is initialized. - // When sut.listPaymentMethods is called below, the cached list will be used. - stubRetrieveCustomers( - key: customerKey, - returningCustomerJSON: expectedCustomerJSON, - expectedCount: 1, - apiClient: apiClient - ) - stubListPaymentMethods( - key: customerKey, - paymentMethodJSONs: expectedPaymentMethodsJSON, - expectedCount: 1, - apiClient: apiClient - ) - - let ekm = MockEphemeralKeyManager(key: customerKey, error: nil) - let sut = STPCustomerContext(keyManager: ekm, apiClient: apiClient) - // Give the mocked API request a little time to complete and cache the customer, then check cache - let exp2 = expectation(description: "listPaymentMethods") - DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + ohhttpDelay) { - sut.listPaymentMethodsForCustomer { paymentMethods, _ in - XCTAssertEqual(paymentMethods!.count, expectedPaymentMethods.count) - exp2.fulfill() - } - } - waitForExpectations(timeout: 2, handler: nil) - } - - func testRetrievePaymentMethodsDoesNotUseCacheIfExpired() { - let customerKey = STPFixtures.ephemeralKey() - let expectedCustomerJSON = STPFixtures.customerWithSingleCardTokenSourceJSON() - let expectedPaymentMethods = [STPFixtures.paymentMethod()] - let expectedPaymentMethodsJSON = [STPFixtures.paymentMethodJSON()] - let apiClient = stubbedAPIClient() - - // apiClient.listPaymentMethods should be called twice: - // - when the context is initialized, - // - when sut.listPaymentMethods is called below, as the cached list has expired. - stubRetrieveCustomers( - key: customerKey, - returningCustomerJSON: expectedCustomerJSON, - expectedCount: 1, - apiClient: apiClient - ) - stubListPaymentMethods( - key: customerKey, - paymentMethodJSONs: expectedPaymentMethodsJSON, - expectedCount: 2, - apiClient: apiClient - ) - - let ekm = MockEphemeralKeyManager(key: customerKey, error: nil) - let sut = STPCustomerContext(keyManager: ekm, apiClient: apiClient) - // Give the mocked API request a little time to complete and cache the customer, then check cache - let exp2 = expectation(description: "listPaymentMethods") - DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + ohhttpDelay) { - sut.paymentMethodsRetrievedDate = Date(timeIntervalSinceNow: -70) - sut.listPaymentMethodsForCustomer { paymentMethods, _ in - XCTAssertEqual(paymentMethods!.count, expectedPaymentMethods.count) - exp2.fulfill() - } - } - waitForExpectations(timeout: 2, handler: nil) - } - - func testRetrievePaymentMethodsDoesNotUseCacheAfterClearingCache() { - let customerKey = STPFixtures.ephemeralKey() - let expectedCustomerJSON = STPFixtures.customerWithSingleCardTokenSourceJSON() - let expectedPaymentMethods = [STPFixtures.paymentMethod()] - let expectedPaymentMethodsJSON = [STPFixtures.paymentMethodJSON()] - let apiClient = stubbedAPIClient() - - // apiClient.listPaymentMethods should be called twice: - // - when the context is initialized, - // - when sut.listPaymentMethods is called below, as the cached list has been cleared - stubRetrieveCustomers( - key: customerKey, - returningCustomerJSON: expectedCustomerJSON, - expectedCount: 1, - apiClient: apiClient - ) - stubListPaymentMethods( - key: customerKey, - paymentMethodJSONs: expectedPaymentMethodsJSON, - expectedCount: 2, - apiClient: apiClient - ) - - let ekm = MockEphemeralKeyManager(key: customerKey, error: nil) - let sut = STPCustomerContext(keyManager: ekm, apiClient: apiClient) - // Give the mocked API request a little time to complete and cache the customer, then check cache - let exp2 = expectation(description: "listPaymentMethods") - DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + ohhttpDelay) { - sut.clearCache() - sut.listPaymentMethodsForCustomer { paymentMethods, _ in - XCTAssertEqual(paymentMethods!.count, expectedPaymentMethods.count) - exp2.fulfill() - } - } - waitForExpectations(timeout: 2, handler: nil) - } - - func testSetCustomerShippingCallsAPIClientCorrectly() { - let address = STPFixtures.address() - let customerKey = STPFixtures.ephemeralKey() - let expectedCustomerJSON = STPFixtures.customerWithSingleCardTokenSourceJSON() - let apiClient = stubbedAPIClient() - - stubRetrieveCustomers( - key: customerKey, - returningCustomerJSON: expectedCustomerJSON, - expectedCount: 1, - apiClient: apiClient - ) - stubListPaymentMethods( - key: customerKey, - paymentMethodJSONs: [], - expectedCount: 1, - apiClient: apiClient - ) - - let exp = expectation(description: "updateCustomer") - stub { urlRequest in - if urlRequest.url?.absoluteString.contains("/customers") ?? false - && urlRequest.httpMethod == "POST" - { - let state = urlRequest.queryItems?.first(where: { item in - item.name == "shipping[address][state]" - })! - XCTAssertEqual(state?.value, address.state) - return true - } - return false - } response: { _ in - exp.fulfill() - return HTTPStubsResponse( - jsonObject: expectedCustomerJSON, - statusCode: 200, - headers: nil - ) - } - - let ekm = MockEphemeralKeyManager(key: customerKey, error: nil) - let sut = STPCustomerContext(keyManager: ekm, apiClient: apiClient) - let exp2 = expectation(description: "updateCustomerWithShipping") - sut.updateCustomer(withShippingAddress: address) { error in - XCTAssertNil(error) - exp2.fulfill() - } - waitForExpectations(timeout: 2, handler: nil) - } - - func testAttachPaymentMethodCallsAPIClientCorrectly() { - let customerKey = STPFixtures.ephemeralKey() - let expectedCustomerJSON = STPFixtures.customerWithSingleCardTokenSourceJSON() - let apiClient = stubbedAPIClient() - let expectedPaymentMethod = STPFixtures.paymentMethod() - let expectedPaymentMethodJSON = STPFixtures.paymentMethodJSON() - let expectedPaymentMethods = [STPFixtures.paymentMethod()] - - stubRetrieveCustomers( - key: customerKey, - returningCustomerJSON: expectedCustomerJSON, - expectedCount: 1, - apiClient: apiClient - ) - stubListPaymentMethods( - key: customerKey, - paymentMethodJSONs: [], - expectedCount: 1, - apiClient: apiClient - ) - - let exp = expectation(description: "payment method attach") - // We're attaching 2 payment methods: - exp.expectedFulfillmentCount = 2 - stub { urlRequest in - if urlRequest.url?.absoluteString.contains("/payment_method") ?? false - && urlRequest.httpMethod == "POST" - { - return true - } - return false - } response: { _ in - exp.fulfill() - return HTTPStubsResponse( - jsonObject: expectedPaymentMethodJSON, - statusCode: 200, - headers: nil - ) - } - - let ekm = MockEphemeralKeyManager(key: customerKey, error: nil) - let sut = STPCustomerContext(keyManager: ekm, apiClient: apiClient) - let exp2 = expectation(description: "CustomerContext attachPaymentMethod") - sut.attachPaymentMethod(toCustomer: expectedPaymentMethods.first!) { error in - XCTAssertNil(error) - exp2.fulfill() - } - - let exp3 = expectation(description: "CustomerContext attachPaymentMethod with ID") - sut.attachPaymentMethodToCustomer(paymentMethodId: expectedPaymentMethod.stripeId) { - error in - XCTAssertNil(error) - exp3.fulfill() - } - - waitForExpectations(timeout: 2, handler: nil) - } - - func testDetachPaymentMethodCallsAPIClientCorrectly() { - let customerKey = STPFixtures.ephemeralKey() - let expectedCustomerJSON = STPFixtures.customerWithSingleCardTokenSourceJSON() - let apiClient = stubbedAPIClient() - let expectedPaymentMethod = STPFixtures.paymentMethod() - let expectedPaymentMethodJSON = STPFixtures.paymentMethodJSON() - let expectedPaymentMethods = [STPFixtures.paymentMethod()] - - stubRetrieveCustomers( - key: customerKey, - returningCustomerJSON: expectedCustomerJSON, - expectedCount: 1, - apiClient: apiClient - ) - stubListPaymentMethods( - key: customerKey, - paymentMethodJSONs: [], - expectedCount: 1, - apiClient: apiClient - ) - - let exp = expectation(description: "payment method detach") - // We're detaching 2 payment methods: - exp.expectedFulfillmentCount = 2 - stub { urlRequest in - if urlRequest.url?.absoluteString.contains("/payment_method") ?? false - && urlRequest.httpMethod == "POST" - { - return true - } - return false - } response: { _ in - exp.fulfill() - return HTTPStubsResponse( - jsonObject: expectedPaymentMethodJSON, - statusCode: 200, - headers: nil - ) - } - - let ekm = MockEphemeralKeyManager(key: customerKey, error: nil) - let sut = STPCustomerContext(keyManager: ekm, apiClient: apiClient) - let exp2 = expectation(description: "CustomerContext detachPaymentMethod") - sut.detachPaymentMethod(fromCustomer: expectedPaymentMethods.first!) { error in - XCTAssertNil(error) - exp2.fulfill() - } - - let exp3 = expectation(description: "CustomerContext detachPaymentMethod with ID") - sut.detachPaymentMethodFromCustomer(paymentMethodId: expectedPaymentMethod.stripeId) { - error in - XCTAssertNil(error) - exp3.fulfill() - } - - waitForExpectations(timeout: 2, handler: nil) - } - - func testFiltersApplePayPaymentMethodsByDefault() { - let customerKey = STPFixtures.ephemeralKey() - let expectedCustomerJSON = STPFixtures.customerWithSingleCardTokenSourceJSON() - let expectedPaymentMethodsJSON = [ - STPFixtures.paymentMethodJSON(), STPFixtures.applePayPaymentMethodJSON(), - ] - let apiClient = stubbedAPIClient() - - stubRetrieveCustomers( - key: customerKey, - returningCustomerJSON: expectedCustomerJSON, - expectedCount: 1, - apiClient: apiClient - ) - stubListPaymentMethods( - key: customerKey, - paymentMethodJSONs: expectedPaymentMethodsJSON, - expectedCount: 1, - apiClient: apiClient - ) - - let ekm = MockEphemeralKeyManager(key: customerKey, error: nil) - let sut = STPCustomerContext(keyManager: ekm, apiClient: apiClient) - // Give the mocked API request a little time to complete and cache the customer, then check cache - let exp2 = expectation(description: "listPaymentMethods") - DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + ohhttpDelay) { - sut.listPaymentMethodsForCustomer { paymentMethods, _ in - // Apple Pay should be filtered out - XCTAssertEqual(paymentMethods!.count, 1) - exp2.fulfill() - } - } - waitForExpectations(timeout: 2, handler: nil) - } - - func testIncludesApplePayPaymentMethods() { - let customerKey = STPFixtures.ephemeralKey() - let expectedCustomerJSON = STPFixtures.customerWithSingleCardTokenSourceJSON() - let expectedPaymentMethodsJSON = [ - STPFixtures.paymentMethodJSON(), STPFixtures.applePayPaymentMethodJSON(), - ] - let apiClient = stubbedAPIClient() - - stubRetrieveCustomers( - key: customerKey, - returningCustomerJSON: expectedCustomerJSON, - expectedCount: 1, - apiClient: apiClient - ) - stubListPaymentMethods( - key: customerKey, - paymentMethodJSONs: expectedPaymentMethodsJSON, - expectedCount: 1, - apiClient: apiClient - ) - - let ekm = MockEphemeralKeyManager(key: customerKey, error: nil) - let sut = STPCustomerContext(keyManager: ekm, apiClient: apiClient) - sut.includeApplePayPaymentMethods = true - // Give the mocked API request a little time to complete and cache the customer, then check cache - let exp2 = expectation(description: "listPaymentMethods") - DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + ohhttpDelay) { - sut.listPaymentMethodsForCustomer { paymentMethods, _ in - // Apple Pay should be included - XCTAssertEqual(paymentMethods!.count, 2) - exp2.fulfill() - } - } - waitForExpectations(timeout: 2, handler: nil) - } - - func testFiltersApplePaySourcesByDefault() { - let customerKey = STPFixtures.ephemeralKey() - let expectedCustomerJSON = STPFixtures.customerWithCardAndApplePaySourcesJSON() - let expectedPaymentMethodsJSON = [ - STPFixtures.paymentMethodJSON(), STPFixtures.applePayPaymentMethodJSON(), - ] - let apiClient = stubbedAPIClient() - - stubRetrieveCustomers( - key: customerKey, - returningCustomerJSON: expectedCustomerJSON, - expectedCount: 1, - apiClient: apiClient - ) - stubListPaymentMethods( - key: customerKey, - paymentMethodJSONs: expectedPaymentMethodsJSON, - expectedCount: 1, - apiClient: apiClient - ) - - let ekm = MockEphemeralKeyManager(key: customerKey, error: nil) - let sut = STPCustomerContext(keyManager: ekm, apiClient: apiClient) - // Give the mocked API request a little time to complete and cache the customer, then check cache - let exp = expectation(description: "retrieveCustomer") - DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + ohhttpDelay) { - sut.retrieveCustomer { customer, _ in - // Apple Pay should be filtered out - XCTAssertEqual(customer!.sources.count, 1) - exp.fulfill() - } - } - waitForExpectations(timeout: 2, handler: nil) - } - - func testIncludeApplePaySources() { - let customerKey = STPFixtures.ephemeralKey() - let expectedCustomerJSON = STPFixtures.customerWithCardAndApplePaySourcesJSON() - let expectedPaymentMethodsJSON = [ - STPFixtures.paymentMethodJSON(), STPFixtures.applePayPaymentMethodJSON(), - ] - let apiClient = stubbedAPIClient() - - stubRetrieveCustomers( - key: customerKey, - returningCustomerJSON: expectedCustomerJSON, - expectedCount: 1, - apiClient: apiClient - ) - stubListPaymentMethods( - key: customerKey, - paymentMethodJSONs: expectedPaymentMethodsJSON, - expectedCount: 1, - apiClient: apiClient - ) - - let ekm = MockEphemeralKeyManager(key: customerKey, error: nil) - let sut = STPCustomerContext(keyManager: ekm, apiClient: apiClient) - sut.includeApplePayPaymentMethods = true - // Give the mocked API request a little time to complete and cache the customer, then check cache - let exp = expectation(description: "retrieveCustomer") - DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + ohhttpDelay) { - sut.retrieveCustomer { customer, _ in - // Apple Pay should be filtered out - XCTAssertEqual(customer!.sources.count, 2) - exp.fulfill() - } - } - waitForExpectations(timeout: 2, handler: nil) - } - - let ohhttpDelay = 0.1 -} diff --git a/Stripe/StripeiOSTests/STPMocks.h b/Stripe/StripeiOSTests/STPMocks.h index 1b782caa656..1b066a680ca 100644 --- a/Stripe/StripeiOSTests/STPMocks.h +++ b/Stripe/StripeiOSTests/STPMocks.h @@ -12,17 +12,6 @@ @interface STPMocks : NSObject -/** - A stateless customer context that always retrieves the same customer object. - */ -+ (STPCustomerContext *)staticCustomerContext; - -/** - A static customer context that always retrieves the given customer and the given payment methods. - Selecting a default source and attaching a source have no effect. - */ -+ (STPCustomerContext *)staticCustomerContextWithCustomer:(STPCustomer *)customer paymentMethods:(NSArray *)paymentMethods; - /** A PaymentConfiguration object with a fake publishable key and a fake apple merchant identifier that ignores the true value of [StripeAPI deviceSupportsApplePay] diff --git a/Stripe/StripeiOSTests/STPMocks.m b/Stripe/StripeiOSTests/STPMocks.m index 63787637994..37cfe4c810f 100644 --- a/Stripe/StripeiOSTests/STPMocks.m +++ b/Stripe/StripeiOSTests/STPMocks.m @@ -26,15 +26,6 @@ - (BOOL)stpmock_applePayEnabled; @implementation STPMocks -+ (STPCustomerContext *)staticCustomerContext { - return [self staticCustomerContextWithCustomer:[STPFixtures customerWithSingleCardTokenSource] - paymentMethods:@[[STPFixtures paymentMethod]]]; -} - -+ (STPCustomerContext *)staticCustomerContextWithCustomer:(STPCustomer *)customer paymentMethods:(NSArray *)paymentMethods { - return [[Testing_StaticCustomerContext_Objc alloc] initWithCustomer:customer paymentMethods:paymentMethods]; -} - + (STPPaymentConfiguration *)paymentConfigurationWithApplePaySupportingDevice { STPPaymentConfiguration *config = [STPPaymentConfiguration new]; config.appleMerchantIdentifier = @"fake_apple_merchant_id"; diff --git a/Stripe/StripeiOSTests/STPPaymentContextApplePayTest.swift b/Stripe/StripeiOSTests/STPPaymentContextApplePayTest.swift deleted file mode 100644 index 1e317ef3d38..00000000000 --- a/Stripe/StripeiOSTests/STPPaymentContextApplePayTest.swift +++ /dev/null @@ -1,204 +0,0 @@ -// -// STPPaymentContextApplePayTest.swift -// StripeiOS Tests -// -// Created by Brian Dorfman on 8/1/16. -// Copyright © 2016 Stripe, Inc. All rights reserved. -// - -@testable@_spi(STP) import Stripe -@testable@_spi(STP) import StripeCore -@testable@_spi(STP) import StripePayments -@testable@_spi(STP) import StripePaymentSheet -@testable@_spi(STP) import StripePaymentsUI - -/// These tests cover STPPaymentContext's Apple Pay specific behavior: -/// - building a PKPaymentRequest -/// - determining paymentSummaryItems -class STPPaymentContextApplePayTest: XCTestCase { - func buildPaymentContext() -> STPPaymentContext { - let config = STPPaymentConfiguration() - config.appleMerchantIdentifier = "fake_merchant_id" - let theme = STPTheme.defaultTheme - let customerContext = Testing_StaticCustomerContext() - let paymentContext = STPPaymentContext( - customerContext: customerContext, - configuration: config, - theme: theme - ) - return paymentContext - } - - // MARK: - buildPaymentRequest - func testBuildPaymentRequest_totalAmount() { - let context = buildPaymentContext() - context.paymentAmount = 150 - let request = context.buildPaymentRequest() - - XCTAssertTrue( - (request?.paymentSummaryItems.last?.amount == NSDecimalNumber(string: "1.50")), - "PKPayment total is not equal to STPPaymentContext amount" - ) - } - - func testBuildPaymentRequest_USDDefault() { - let context = buildPaymentContext() - context.paymentAmount = 100 - let request = context.buildPaymentRequest() - - XCTAssertTrue( - (request?.currencyCode == "USD"), - "Default PKPaymentRequest currency code is not USD" - ) - } - - func testBuildPaymentRequest_currency() { - let context = buildPaymentContext() - context.paymentAmount = 100 - context.paymentCurrency = "GBP" - let request = context.buildPaymentRequest() - - XCTAssertTrue( - (request?.currencyCode == "GBP"), - "PKPaymentRequest currency code is not equal to STPPaymentContext currency" - ) - } - - func testBuildPaymentRequest_uppercaseCurrency() { - let context = buildPaymentContext() - context.paymentAmount = 100 - context.paymentCurrency = "eur" - let request = context.buildPaymentRequest() - - XCTAssertTrue( - (request?.currencyCode == "EUR"), - "PKPaymentRequest currency code is not uppercased" - ) - } - - func testSummaryItems() -> [PKPaymentSummaryItem]? { - return [ - PKPaymentSummaryItem( - label: "First item", - amount: NSDecimalNumber(mantissa: 20, exponent: 0, isNegative: false) - ), - PKPaymentSummaryItem( - label: "Second item", - amount: NSDecimalNumber(mantissa: 90, exponent: 0, isNegative: false) - ), - PKPaymentSummaryItem( - label: "Discount", - amount: NSDecimalNumber(mantissa: 10, exponent: 0, isNegative: true) - ), - PKPaymentSummaryItem( - label: "Total", - amount: NSDecimalNumber(mantissa: 100, exponent: 0, isNegative: false) - ), - ] - } - - func testBuildPaymentRequest_summaryItems() { - let context = buildPaymentContext() - context.paymentSummaryItems = testSummaryItems()! - let request = context.buildPaymentRequest() - - XCTAssertTrue((request?.paymentSummaryItems == context.paymentSummaryItems)) - } - - // MARK: - paymentSummaryItems - func testSetPaymentAmount_generateSummaryItems() { - let context = buildPaymentContext() - context.paymentAmount = 10000 - context.paymentCurrency = "USD" - let itemTotalAmount = context.paymentSummaryItems.last?.amount - let correctTotalAmount = NSDecimalNumber.stp_decimalNumber( - withAmount: context.paymentAmount, - currency: context.paymentCurrency - ) - - XCTAssertTrue((itemTotalAmount == correctTotalAmount)) - } - - func testSetPaymentAmount_generateSummaryItemsShippingMethod() { - let context = buildPaymentContext() - context.paymentAmount = 100 - context.configuration.companyName = "Foo Company" - let method = PKShippingMethod() - method.amount = NSDecimalNumber(string: "5.99") - method.label = "FedEx" - method.detail = "foo" - method.identifier = "123" - context.selectedShippingMethod = method - - let items = context.paymentSummaryItems - XCTAssertEqual(Int(items.count), 2) - let item1 = items[0] - XCTAssertEqual(item1.label, "FedEx") - XCTAssertEqual(item1.amount, NSDecimalNumber(string: "5.99")) - let item2 = items[1] - XCTAssertEqual(item2.label, "Foo Company") - XCTAssertEqual(item2.amount, NSDecimalNumber(string: "6.99")) - } - - func testSummaryItemsToSummaryItems_shippingMethod() { - let context = buildPaymentContext() - let item1 = PKPaymentSummaryItem() - item1.amount = NSDecimalNumber(string: "1.00") - item1.label = "foo" - let item2 = PKPaymentSummaryItem() - item2.amount = NSDecimalNumber(string: "9.00") - item2.label = "bar" - let item3 = PKPaymentSummaryItem() - item3.amount = NSDecimalNumber(string: "10.00") - item3.label = "baz" - context.paymentSummaryItems = [item1, item2, item3] - let method = PKShippingMethod() - method.amount = NSDecimalNumber(string: "5.99") - method.label = "FedEx" - method.detail = "foo" - method.identifier = "123" - context.selectedShippingMethod = method - - let items = context.paymentSummaryItems - XCTAssertEqual(Int(items.count), 4) - let resultItem1 = items[0] - XCTAssertEqual(resultItem1.label, "foo") - XCTAssertEqual(resultItem1.amount, NSDecimalNumber(string: "1.00")) - let resultItem2 = items[1] - XCTAssertEqual(resultItem2.label, "bar") - XCTAssertEqual(resultItem2.amount, NSDecimalNumber(string: "9.00")) - let resultItem3 = items[2] - XCTAssertEqual(resultItem3.label, "FedEx") - XCTAssertEqual(resultItem3.amount, NSDecimalNumber(string: "5.99")) - let resultItem4 = items[3] - XCTAssertEqual(resultItem4.label, "baz") - XCTAssertEqual(resultItem4.amount, NSDecimalNumber(string: "15.99")) - } - - func testAmountToAmount_shippingMethod_usd() { - let context = buildPaymentContext() - context.paymentAmount = 100 - let method = PKShippingMethod() - method.amount = NSDecimalNumber(string: "5.99") - method.label = "FedEx" - method.detail = "foo" - method.identifier = "123" - context.selectedShippingMethod = method - let amount = context.paymentAmount - XCTAssertEqual(amount, 699) - } - - func testSummaryItems_generateAmountDecimalCurrency() { - let context = buildPaymentContext() - context.paymentSummaryItems = testSummaryItems()! - context.paymentCurrency = "USD" - XCTAssertTrue(context.paymentAmount == 10000) - } - - func testSummaryItems_generateAmountNoDecimalCurrency() { - let context = buildPaymentContext() - context.paymentSummaryItems = testSummaryItems()! - context.paymentCurrency = "JPY" - XCTAssertTrue(context.paymentAmount == 100) - } -} diff --git a/Stripe/StripeiOSTests/STPPaymentContextSnapshotTests.swift b/Stripe/StripeiOSTests/STPPaymentContextSnapshotTests.swift deleted file mode 100644 index 319e9b9673c..00000000000 --- a/Stripe/StripeiOSTests/STPPaymentContextSnapshotTests.swift +++ /dev/null @@ -1,84 +0,0 @@ -// Converted to Swift 5.8.1 by Swiftify v5.8.28463 - https://swiftify.com/ -// -// STPPaymentContextSnapshotTests.m -// StripeiOS Tests -// -// Created by Ben Guo on 12/13/17. -// Copyright © 2017 Stripe, Inc. All rights reserved. -// - -import iOSSnapshotTestCaseCore -import StripeCoreTestUtils - -class STPPaymentContextSnapshotTests: STPSnapshotTestCase { - var customerContext: STPCustomerContext? - var config: STPPaymentConfiguration? - var hostViewController: UINavigationController? - var paymentContext: STPPaymentContext? - - override func setUp() { - super.setUp() - let config = STPPaymentConfiguration() - config.companyName = "Test Company" - config.requiredBillingAddressFields = .full - config.shippingType = .shipping - self.config = config - let customerContext = Testing_StaticCustomerContext_Objc.init(customer: STPFixtures.customerWithCardTokenAndSourceSources(), paymentMethods: [STPFixtures.paymentMethod(), STPFixtures.paymentMethod()]) - self.customerContext = customerContext - - let viewController = UIViewController() - hostViewController = stp_navigationControllerForSnapshotTest(withRootVC: viewController) - } - - func buildPaymentContext() { - let context = STPPaymentContext(customerContext: customerContext!) - context.hostViewController = hostViewController - context.configuration.requiredShippingAddressFields = Set([STPContactField.emailAddress]) - paymentContext = context - } - - func testPushPaymentOptionsSmallTitle() { - buildPaymentContext() - - hostViewController?.navigationBar.prefersLargeTitles = false - paymentContext?.largeTitleDisplayMode = UINavigationItem.LargeTitleDisplayMode.automatic - paymentContext?.pushPaymentOptionsViewController() - let view = stp_preparedAndSizedViewForSnapshotTest(from: hostViewController)! - STPSnapshotVerifyView(view, identifier: nil) - } - - // This test renders at a slightly larger size half the time. - // We're deprecating Basic Integration soon, and we've spent enough time on this, - // so these tests are being disabled for now. - // - (void)testPushPaymentOptionsLargeTitle { - // [self buildPaymentContext]; - // - // self.hostViewController.navigationBar.prefersLargeTitles = YES; - // self.paymentContext.largeTitleDisplayMode = UINavigationItemLargeTitleDisplayModeAutomatic; - // [self.paymentContext pushPaymentOptionsViewController]; - // UIView *view = [self stp_preparedAndSizedViewForSnapshotTestFromNavigationController:self.hostViewController]; - // STPSnapshotVerifyView(view, nil); - // } - - func testPushShippingAddressSmallTitle() { - buildPaymentContext() - - hostViewController?.navigationBar.prefersLargeTitles = false - paymentContext?.largeTitleDisplayMode = UINavigationItem.LargeTitleDisplayMode.automatic - paymentContext?.pushShippingViewController() - let view = stp_preparedAndSizedViewForSnapshotTest(from: hostViewController)! - STPSnapshotVerifyView(view, identifier: nil) - } - // This test renders at a slightly larger size half the time. - // We're deprecating Basic Integration soon, and we've spent enough time on this, - // so these tests are being disabled for now. - // - (void)testPushShippingAddressLargeTitle { - // [self buildPaymentContext]; - // - // self.hostViewController.navigationBar.prefersLargeTitles = YES; - // self.paymentContext.largeTitleDisplayMode = UINavigationItemLargeTitleDisplayModeAutomatic; - // [self.paymentContext pushShippingViewController]; - // UIView *view = [self stp_preparedAndSizedViewForSnapshotTestFromNavigationController:self.hostViewController]; - // STPSnapshotVerifyView(view, nil); - // } -} diff --git a/Stripe/StripeiOSTests/STPPaymentOptionsViewControllerLocalizationSnapshotTests.swift b/Stripe/StripeiOSTests/STPPaymentOptionsViewControllerLocalizationSnapshotTests.swift deleted file mode 100644 index 0853069526d..00000000000 --- a/Stripe/StripeiOSTests/STPPaymentOptionsViewControllerLocalizationSnapshotTests.swift +++ /dev/null @@ -1,102 +0,0 @@ -// -// STPPaymentOptionsViewControllerLocalizationSnapshotTests.swift -// StripeiOS Tests -// -// Created by Brian Dorfman on 10/17/16. -// Copyright © 2016 Stripe, Inc. All rights reserved. -// - -import iOSSnapshotTestCase -import StripeCoreTestUtils - -@testable@_spi(STP) import Stripe -@testable@_spi(STP) import StripeCore -@testable@_spi(STP) import StripePayments -@testable@_spi(STP) import StripePaymentSheet -@testable@_spi(STP) import StripePaymentsUI - -class MockSTPPaymentOptionsViewControllerDelegate: NSObject, STPPaymentOptionsViewControllerDelegate -{ - func paymentOptionsViewController( - _ paymentOptionsViewController: STPPaymentOptionsViewController, - didFailToLoadWithError error: Error - ) { - } - - func paymentOptionsViewControllerDidFinish( - _ paymentOptionsViewController: STPPaymentOptionsViewController - ) { - } - - func paymentOptionsViewControllerDidCancel( - _ paymentOptionsViewController: STPPaymentOptionsViewController - ) { - } - -} - -class STPPaymentOptionsViewControllerLocalizationSnapshotTests: STPSnapshotTestCase { - - func performSnapshotTest(forLanguage language: String?) { - let config = STPPaymentConfiguration() - config.companyName = "Test Company" - config.requiredBillingAddressFields = .full - let theme = STPTheme.defaultTheme - let paymentMethods = [STPFixtures.paymentMethod(), STPFixtures.paymentMethod()] - let customerContext = Testing_StaticCustomerContext.init( - customer: STPFixtures.customerWithCardTokenAndSourceSources(), - paymentMethods: paymentMethods - ) - let delegate = MockSTPPaymentOptionsViewControllerDelegate() - STPLocalizationUtils.overrideLanguage(to: language) - let paymentOptionsVC = STPPaymentOptionsViewController( - configuration: config, - theme: theme, - customerContext: customerContext, - delegate: delegate - ) - let didLoadExpectation = expectation(description: "VC did load") - - paymentOptionsVC.loadingPromise?.onSuccess({ (_) in - didLoadExpectation.fulfill() - }) - wait(for: [didLoadExpectation].compactMap { $0 }, timeout: 2) - - let viewToTest = stp_preparedAndSizedViewForSnapshotTest(from: paymentOptionsVC)! - - STPSnapshotVerifyView(viewToTest, identifier: nil) - STPLocalizationUtils.overrideLanguage(to: nil) - } - - func testGerman() { - performSnapshotTest(forLanguage: "de") - } - - func testEnglish() { - performSnapshotTest(forLanguage: "en") - } - - func testSpanish() { - performSnapshotTest(forLanguage: "es") - } - - func testFrench() { - performSnapshotTest(forLanguage: "fr") - } - - func testItalian() { - performSnapshotTest(forLanguage: "it") - } - - func testJapanese() { - performSnapshotTest(forLanguage: "ja") - } - - func testDutch() { - performSnapshotTest(forLanguage: "nl") - } - - func testChinese() { - performSnapshotTest(forLanguage: "zh-Hans") - } -} diff --git a/Stripe/StripeiOSTests/STPPaymentOptionsViewControllerTest.swift b/Stripe/StripeiOSTests/STPPaymentOptionsViewControllerTest.swift deleted file mode 100644 index 0555fa9799a..00000000000 --- a/Stripe/StripeiOSTests/STPPaymentOptionsViewControllerTest.swift +++ /dev/null @@ -1,351 +0,0 @@ -// -// STPPaymentOptionsViewControllerTest.swift -// StripeiOS Tests -// -// Created by Brian Dorfman on 10/10/17. -// Copyright © 2017 Stripe, Inc. All rights reserved. -// - -import OCMock - -@testable@_spi(STP) import Stripe -@testable@_spi(STP) import StripeCore -@testable@_spi(STP) import StripePayments -@testable@_spi(STP) import StripePaymentSheet -@testable@_spi(STP) import StripePaymentsUI - -class STPPaymentOptionsViewControllerTest: XCTestCase { - class MockSTPPaymentOptionsViewControllerDelegate: NSObject, - STPPaymentOptionsViewControllerDelegate - { - var didFail = false - func paymentOptionsViewController( - _ paymentOptionsViewController: STPPaymentOptionsViewController, - didFailToLoadWithError error: Error - ) { - didFail = true - } - - var didFinish = false - func paymentOptionsViewControllerDidFinish( - _ paymentOptionsViewController: STPPaymentOptionsViewController - ) { - didFinish = true - } - - var didCancel = false - func paymentOptionsViewControllerDidCancel( - _ paymentOptionsViewController: STPPaymentOptionsViewController - ) { - didCancel = true - } - - var didSelect = false - func paymentOptionsViewController( - _ paymentOptionsViewController: STPPaymentOptionsViewController, - didSelect paymentOption: STPPaymentOption - ) { - didSelect = true - } - } - - func buildViewController( - with customer: STPCustomer, - paymentMethods: [STPPaymentMethod], - configuration config: STPPaymentConfiguration, - delegate: STPPaymentOptionsViewControllerDelegate - ) -> STPPaymentOptionsViewController { - let mockCustomerContext = Testing_StaticCustomerContext( - customer: customer, - paymentMethods: paymentMethods - ) - return buildViewController( - with: mockCustomerContext, - configuration: config, - delegate: delegate - ) - } - - func buildViewController( - with customerContext: STPCustomerContext, - configuration config: STPPaymentConfiguration, - delegate: STPPaymentOptionsViewControllerDelegate - ) -> STPPaymentOptionsViewController { - let theme = STPTheme.defaultTheme - let vc = STPPaymentOptionsViewController( - configuration: config, - theme: theme, - customerContext: customerContext, - delegate: delegate - ) - let didLoadExpectation = expectation(description: "VC did load") - vc.loadingPromise?.onSuccess({ (_) in - didLoadExpectation.fulfill() - }) - - wait(for: [didLoadExpectation], timeout: 2) - - return vc - } - - /// When the customer has no sources, and card is the sole available payment - /// method, STPAddCardViewController should be shown. - func testInitWithNoSourcesAndConfigWithUseSourcesOffAndCardAvailable() { - let customer = STPFixtures.customerWithNoSources() - let config = STPPaymentConfiguration() - config.applePayEnabled = false - let delegate = MockSTPPaymentOptionsViewControllerDelegate() - let sut = buildViewController( - with: customer, - paymentMethods: [], - configuration: config, - delegate: delegate - ) - XCTAssertTrue((sut.internalViewController is STPAddCardViewController)) - } - - /// When the customer has a single card token source and the available payment methods - /// are card and apple pay, STPPaymentOptionsInternalVC should be shown. - func testInitWithSingleCardTokenSourceAndCardAvailable() { - let customer = STPFixtures.customerWithSingleCardTokenSource() - let paymentMethods = [STPFixtures.paymentMethod()] - let config = STPPaymentConfiguration() - let delegate = MockSTPPaymentOptionsViewControllerDelegate() - let sut = buildViewController( - with: customer, - paymentMethods: paymentMethods.compactMap { $0 }, - configuration: config, - delegate: delegate - ) - XCTAssertTrue((sut.internalViewController is STPPaymentOptionsInternalViewController)) - } - - /// When the customer has a single card source source and the available payment methods - /// are card only, STPPaymentOptionsInternalVC should be shown. - func testInitWithSingleCardSourceSourceAndCardAvailable() { - let customer = STPFixtures.customerWithSingleCardSourceSource() - let paymentMethods = [STPFixtures.paymentMethod()] - let config = STPPaymentConfiguration() - config.applePayEnabled = false - let delegate = MockSTPPaymentOptionsViewControllerDelegate() - let sut = buildViewController( - with: customer, - paymentMethods: paymentMethods.compactMap { $0 }, - configuration: config, - delegate: delegate - ) - XCTAssertTrue((sut.internalViewController is STPPaymentOptionsInternalViewController)) - } - - /// Tapping cancel in an internal AddCard view controller should result in a call to - /// didCancel: - func testAddCardCancelForwardsToDelegate() { - let customer = STPFixtures.customerWithNoSources() - let config = STPPaymentConfiguration() - let delegate = MockSTPPaymentOptionsViewControllerDelegate() - let sut = buildViewController( - with: customer, - paymentMethods: [], - configuration: config, - delegate: delegate - ) - XCTAssertTrue((sut.internalViewController is STPAddCardViewController)) - let cancelButton = sut.internalViewController?.navigationItem.leftBarButtonItem - _ = cancelButton?.target?.perform(cancelButton?.action, with: cancelButton) - - XCTAssertTrue(delegate.didCancel) - } - - /// Tapping cancel in an internal PaymentOptionsInternal view controller should - /// result in a call to didCancel: - func testInternalCancelForwardsToDelegate() { - let customer = STPFixtures.customerWithSingleCardTokenSource() - let paymentMethods = [STPFixtures.paymentMethod()] - let config = STPPaymentConfiguration() - let delegate = MockSTPPaymentOptionsViewControllerDelegate() - let sut = buildViewController( - with: customer, - paymentMethods: paymentMethods.compactMap { $0 }, - configuration: config, - delegate: delegate - ) - XCTAssertTrue((sut.internalViewController is STPPaymentOptionsInternalViewController)) - let cancelButton = sut.internalViewController?.navigationItem.leftBarButtonItem - _ = cancelButton?.target?.perform(cancelButton?.action, with: cancelButton) - - XCTAssertTrue(delegate.didCancel) - } - - /// When an AddCard view controller creates a card payment method, it should be attached to the - /// customer and the correct delegate methods should be called. - func testAddCardAttachesToCustomerAndFinishes() { - let config = STPPaymentConfiguration() - let customer = STPFixtures.customerWithNoSources() - let mockCustomerContext = Testing_StaticCustomerContext( - customer: customer, - paymentMethods: [] - ) - let delegate = MockSTPPaymentOptionsViewControllerDelegate() - let sut = buildViewController( - with: mockCustomerContext, - configuration: config, - delegate: delegate - ) - XCTAssertNotNil(sut.view) - XCTAssertTrue((sut.internalViewController is STPAddCardViewController)) - - let internalVC = sut.internalViewController as? STPAddCardViewController - let exp = expectation(description: "completion") - let expectedPaymentMethod = STPFixtures.paymentMethod() - internalVC?.delegate?.addCardViewController( - internalVC!, - didCreatePaymentMethod: expectedPaymentMethod - ) { error in - XCTAssertNil(error) - exp.fulfill() - } - - let _: ((Any?) -> Bool)? = { obj in - let paymentMethod = obj as? STPPaymentMethod - return paymentMethod?.stripeId == expectedPaymentMethod.stripeId - } - XCTAssertTrue(mockCustomerContext.didAttach) - XCTAssertTrue(delegate.didSelect) - XCTAssertTrue(delegate.didFinish) - waitForExpectations(timeout: 2, handler: nil) - } - - // Tests for race condition where the promise for fetching payment methods - // finishes in the context of intializing the sut, and `addCardViewControllerFooterView` - // is set directly after init, while internalViewController is `STPAddCardViewController` - func testSetAfterInit_addCardViewControllerFooterView_STPAddCardViewController() { - let customer = STPFixtures.customerWithNoSources() - let config = STPPaymentConfiguration() - config.applePayEnabled = false - let delegate = MockSTPPaymentOptionsViewControllerDelegate() - let sut = buildViewController( - with: customer, - paymentMethods: [], - configuration: config, - delegate: delegate - ) - sut.addCardViewControllerFooterView = UIView() - guard let payMethodsInternal = sut.internalViewController as? STPAddCardViewController else { - XCTFail() - return - } - XCTAssertNotNil(payMethodsInternal.customFooterView) - } - - // Tests for race condition where the promise for fetching payment methods - // finishes in the context of intializing the sut, and the `paymentOptionsViewControllerFooterView` - // is set directly after init, while internalViewController is `STPPaymentOptionsInternalViewController` - func testSetAfterInit_paymentOptionsViewControllerFooterView_STPPaymentOptionsInternalViewController() { - let customer = STPFixtures.customerWithSingleCardTokenSource() - let paymentMethods = [STPFixtures.paymentMethod()] - let config = STPPaymentConfiguration() - let delegate = MockSTPPaymentOptionsViewControllerDelegate() - let sut = buildViewController( - with: customer, - paymentMethods: paymentMethods.compactMap { $0 }, - configuration: config, - delegate: delegate - ) - sut.paymentOptionsViewControllerFooterView = UIView() - guard let payMethodsInternal = sut.internalViewController as? STPPaymentOptionsInternalViewController else { - XCTFail() - return - } -#if compiler(>=5.7) - XCTAssertNotNil(payMethodsInternal.customFooterView) -#endif - } - - // Tests for race condition where the promise for fetching payment methods - // finishes in the context of init the sut, and the `addCardViewControllerFooterView` - // is set directly after init, while internalViewController is `STPPaymentOptionsInternalViewController` - func testSetAfterInit_addCardViewControllerFooterView_STPPaymentOptionsInternalViewController() { - let customer = STPFixtures.customerWithSingleCardTokenSource() - let paymentMethods = [STPFixtures.paymentMethod()] - let config = STPPaymentConfiguration() - let delegate = MockSTPPaymentOptionsViewControllerDelegate() - let sut = buildViewController( - with: customer, - paymentMethods: paymentMethods.compactMap { $0 }, - configuration: config, - delegate: delegate - ) - sut.addCardViewControllerFooterView = UIView() - guard let payMethodsInternal = sut.internalViewController as? STPPaymentOptionsInternalViewController else { - XCTFail() - return - } -#if compiler(>=5.7) - XCTAssertNotNil(payMethodsInternal.addCardViewControllerCustomFooterView) -#endif - } - - // Tests for race condition where the promise for fetching payment methods - // finishes in the context of init the sut, and the `prefilledInformation` - // is set directly after init, while internalViewController is `STPPaymentOptionsInternalViewController` - func testSetAfterInit_prefilledInformation_STPPaymentOptionsInternalViewController() { - let customer = STPFixtures.customerWithSingleCardTokenSource() - let paymentMethods = [STPFixtures.paymentMethod()] - let config = STPPaymentConfiguration() - let delegate = MockSTPPaymentOptionsViewControllerDelegate() - let sut = buildViewController( - with: customer, - paymentMethods: paymentMethods.compactMap { $0 }, - configuration: config, - delegate: delegate - ) - let userInformation = STPUserInformation() - let address = STPAddress() - address.name = "John Doe" - address.line1 = "123 Main" - address.city = "Seattle" - address.state = "Washington" - address.postalCode = "98104" - address.phone = "2065551234" - userInformation.billingAddress = address - sut.prefilledInformation = userInformation - guard let payMethodsInternal = sut.internalViewController as? STPPaymentOptionsInternalViewController else { - XCTFail() - return - } -#if compiler(>=5.7) - XCTAssertNotNil(payMethodsInternal.prefilledInformation) -#endif - } - - // Tests for race condition where the promise for fetching payment methods - // finishes in the context of init the sut, and the `prefilledInformation` - // is set directly after init, while internalViewController is `STPAddCardViewController` - func testSetAfterInit_prefilledInformation_STPAddCardViewController() { - let customer = STPFixtures.customerWithNoSources() - let config = STPPaymentConfiguration() - config.applePayEnabled = false - let delegate = MockSTPPaymentOptionsViewControllerDelegate() - let sut = buildViewController( - with: customer, - paymentMethods: [], - configuration: config, - delegate: delegate - ) - let userInformation = STPUserInformation() - let address = STPAddress() - address.name = "John Doe" - address.line1 = "123 Main" - address.city = "Seattle" - address.state = "Washington" - address.postalCode = "98104" - address.phone = "2065551234" - userInformation.billingAddress = address - sut.prefilledInformation = userInformation - guard let payMethodsInternal = sut.internalViewController as? STPAddCardViewController else { - XCTFail() - return - } - XCTAssertNotNil(payMethodsInternal.prefilledInformation) - } -} diff --git a/Stripe/StripeiOSTests/STPShippingAddressViewControllerLocalizationSnapshotTests.swift b/Stripe/StripeiOSTests/STPShippingAddressViewControllerLocalizationSnapshotTests.swift deleted file mode 100644 index 1760b7a0d41..00000000000 --- a/Stripe/StripeiOSTests/STPShippingAddressViewControllerLocalizationSnapshotTests.swift +++ /dev/null @@ -1,113 +0,0 @@ -// -// STPShippingAddressViewControllerLocalizationSnapshotTests.swift -// StripeiOS Tests -// -// Created by Ben Guo on 11/3/16. -// Copyright © 2016 Stripe, Inc. All rights reserved. -// - -import iOSSnapshotTestCase -import StripeCoreTestUtils - -@testable@_spi(STP) import Stripe -@testable@_spi(STP) import StripeCore -@testable@_spi(STP) import StripePayments -@testable@_spi(STP) import StripePaymentSheet -@testable@_spi(STP) import StripePaymentsUI - -class STPShippingAddressViewControllerLocalizationSnapshotTests: STPSnapshotTestCase { - - func performSnapshotTest( - forLanguage language: String?, - shippingType: STPShippingType, - contact: Bool - ) { - var identifier = (shippingType == .shipping) ? "shipping" : "delivery" - let config = STPPaymentConfiguration() - config.companyName = "Test Company" - config.requiredShippingAddressFields = Set([ - .postalAddress, - .emailAddress, - .phoneNumber, - .name, - ]) - if contact { - config.requiredShippingAddressFields = Set([.emailAddress]) - identifier = "contact" - } - config.shippingType = shippingType - - STPLocalizationUtils.overrideLanguage(to: language) - let info = STPUserInformation() - info.billingAddress = STPAddress() - info.billingAddress!.email = "@" // trigger "use billing address" button - - let shippingVC = STPShippingAddressViewController( - configuration: config, - theme: STPTheme.defaultTheme, - currency: nil, - shippingAddress: nil, - selectedShippingMethod: nil, - prefilledInformation: info - ) - - /// This method rejects nil or empty country codes to stop strange looking behavior - /// when scrolling to the top "unset" position in the picker, so put in - /// an invalid country code instead to test seeing the "Country" placeholder - shippingVC.addressViewModel.addressFieldTableViewCountryCode = "INVALID" - - let viewToTest = stp_preparedAndSizedViewForSnapshotTest(from: shippingVC)! - - STPSnapshotVerifyView(viewToTest, identifier: identifier) - - STPLocalizationUtils.overrideLanguage(to: nil) - } - - func testGerman() { - performSnapshotTest(forLanguage: "de", shippingType: .shipping, contact: false) - performSnapshotTest(forLanguage: "de", shippingType: .shipping, contact: true) - performSnapshotTest(forLanguage: "de", shippingType: .delivery, contact: false) - } - - func testEnglish() { - performSnapshotTest(forLanguage: "en", shippingType: .shipping, contact: false) - performSnapshotTest(forLanguage: "en", shippingType: .shipping, contact: true) - performSnapshotTest(forLanguage: "en", shippingType: .delivery, contact: false) - } - - func testSpanish() { - performSnapshotTest(forLanguage: "es", shippingType: .shipping, contact: false) - performSnapshotTest(forLanguage: "es", shippingType: .shipping, contact: true) - performSnapshotTest(forLanguage: "es", shippingType: .delivery, contact: false) - } - - func testFrench() { - performSnapshotTest(forLanguage: "fr", shippingType: .shipping, contact: false) - performSnapshotTest(forLanguage: "fr", shippingType: .shipping, contact: true) - performSnapshotTest(forLanguage: "fr", shippingType: .delivery, contact: false) - } - - func testItalian() { - performSnapshotTest(forLanguage: "it", shippingType: .shipping, contact: false) - performSnapshotTest(forLanguage: "it", shippingType: .shipping, contact: true) - performSnapshotTest(forLanguage: "it", shippingType: .delivery, contact: false) - } - - func testJapanese() { - performSnapshotTest(forLanguage: "ja", shippingType: .shipping, contact: false) - performSnapshotTest(forLanguage: "ja", shippingType: .shipping, contact: true) - performSnapshotTest(forLanguage: "ja", shippingType: .delivery, contact: false) - } - - func testDutch() { - performSnapshotTest(forLanguage: "nl", shippingType: .shipping, contact: false) - performSnapshotTest(forLanguage: "nl", shippingType: .shipping, contact: true) - performSnapshotTest(forLanguage: "nl", shippingType: .delivery, contact: false) - } - - func testChinese() { - performSnapshotTest(forLanguage: "zh-Hans", shippingType: .shipping, contact: false) - performSnapshotTest(forLanguage: "zh-Hans", shippingType: .shipping, contact: true) - performSnapshotTest(forLanguage: "zh-Hans", shippingType: .delivery, contact: false) - } -} diff --git a/Stripe/StripeiOSTests/STPShippingAddressViewControllerTest.swift b/Stripe/StripeiOSTests/STPShippingAddressViewControllerTest.swift deleted file mode 100644 index cc473049d34..00000000000 --- a/Stripe/StripeiOSTests/STPShippingAddressViewControllerTest.swift +++ /dev/null @@ -1,114 +0,0 @@ -// -// STPShippingAddressViewControllerTest.swift -// StripeiOS Tests -// -// Created by Cameron Sabol on 8/7/18. -// Copyright © 2018 Stripe, Inc. All rights reserved. -// - -import Stripe - -@testable@_spi(STP) import Stripe -@testable@_spi(STP) import StripeCore -@testable@_spi(STP) import StripePayments -@testable@_spi(STP) import StripePaymentSheet -@testable@_spi(STP) import StripePaymentsUI - -class STPShippingAddressViewControllerTest: XCTestCase { - func testPrefilledBillingAddress_removeAddress() { - let config = STPPaymentConfiguration() - config.requiredShippingAddressFields = Set([.postalAddress]) - - let address = STPAddress() - address.name = "John Smith Doe" - address.phone = "8885551212" - address.email = "foo@example.com" - address.line1 = "55 John St" - address.city = "Harare" - address.postalCode = "10002" - // Zimbabwe does not require zip codes, while the default locale for tests (US) does - address.country = "ZW" - // Sanity checks - XCTAssertFalse(STPPostalCodeValidator.postalCodeIsRequired(forCountryCode: "ZW")) - XCTAssertTrue(STPPostalCodeValidator.postalCodeIsRequired(forCountryCode: "US")) - - let sut = STPShippingAddressViewController( - configuration: config, - theme: STPTheme.defaultTheme, - currency: nil, - shippingAddress: address, - selectedShippingMethod: nil, - prefilledInformation: nil - ) - - XCTAssertNoThrow(sut.loadView()) - XCTAssertNoThrow(sut.viewDidLoad()) - } - - func testPrefilledBillingAddress_addAddressWithLimitedCountries() { - // Zimbabwe does not require zip codes, while the default locale for tests (US) does - NSLocale.stp_withLocale(as: NSLocale(localeIdentifier: "en_ZW")) { - // Sanity checks - XCTAssertFalse(STPPostalCodeValidator.postalCodeIsRequired(forCountryCode: "ZW")) - XCTAssertTrue(STPPostalCodeValidator.postalCodeIsRequired(forCountryCode: "US")) - let config = STPPaymentConfiguration() - config.requiredShippingAddressFields = Set([.postalAddress]) - config.availableCountries = Set(["CA", "BT"]) - - let address = STPAddress() - address.name = "John Smith Doe" - address.phone = "8885551212" - address.email = "foo@example.com" - address.line1 = "55 John St" - address.city = "New York" - address.state = "NY" - address.postalCode = "10002" - address.country = "US" - - let sut = STPShippingAddressViewController( - configuration: config, - theme: STPTheme.defaultTheme, - currency: nil, - shippingAddress: address, - selectedShippingMethod: nil, - prefilledInformation: nil - ) - - XCTAssertNoThrow(sut.loadView()) - XCTAssertNoThrow(sut.viewDidLoad()) - } - } - - func testPrefilledBillingAddress_addAddress() { - // Zimbabwe does not require zip codes, while the default locale for tests (US) does - NSLocale.stp_withLocale(as: NSLocale(localeIdentifier: "en_ZW")) { - // Sanity checks - XCTAssertFalse(STPPostalCodeValidator.postalCodeIsRequired(forCountryCode: "ZW")) - XCTAssertTrue(STPPostalCodeValidator.postalCodeIsRequired(forCountryCode: "US")) - let config = STPPaymentConfiguration() - config.requiredShippingAddressFields = Set([.postalAddress]) - - let address = STPAddress() - address.name = "John Smith Doe" - address.phone = "8885551212" - address.email = "foo@example.com" - address.line1 = "55 John St" - address.city = "New York" - address.state = "NY" - address.postalCode = "10002" - address.country = "US" - - let sut = STPShippingAddressViewController( - configuration: config, - theme: STPTheme.defaultTheme, - currency: nil, - shippingAddress: address, - selectedShippingMethod: nil, - prefilledInformation: nil - ) - - XCTAssertNoThrow(sut.loadView()) - XCTAssertNoThrow(sut.viewDidLoad()) - } - } -} diff --git a/Stripe/StripeiOSTests/STPShippingMethodsViewControllerLocalizationSnapshotTests.swift b/Stripe/StripeiOSTests/STPShippingMethodsViewControllerLocalizationSnapshotTests.swift deleted file mode 100644 index f9b256b34ee..00000000000 --- a/Stripe/StripeiOSTests/STPShippingMethodsViewControllerLocalizationSnapshotTests.swift +++ /dev/null @@ -1,76 +0,0 @@ -// -// STPShippingMethodsViewControllerLocalizationSnapshotTests.swift -// StripeiOS Tests -// -// Created by Ben Guo on 11/3/16. -// Copyright © 2016 Stripe, Inc. All rights reserved. -// - -import iOSSnapshotTestCase -import StripeCoreTestUtils - -@testable@_spi(STP) import Stripe -@testable@_spi(STP) import StripeCore -@testable@_spi(STP) import StripePayments -@testable@_spi(STP) import StripePaymentSheet -@testable@_spi(STP) import StripePaymentsUI - -class STPShippingMethodsViewControllerLocalizationSnapshotTests: STPSnapshotTestCase { - - func performSnapshotTest(forLanguage language: String?) { - STPLocalizationUtils.overrideLanguage(to: language) - - let method1 = PKShippingMethod() - method1.label = "UPS Ground" - method1.detail = "Arrives in 3-5 days" - method1.amount = NSDecimalNumber(string: "0.00") - method1.identifier = "ups_ground" - let method2 = PKShippingMethod() - method2.label = "FedEx" - method2.detail = "Arrives tomorrow" - method2.amount = NSDecimalNumber(string: "5.99") - method2.identifier = "fedex" - - let shippingVC = STPShippingMethodsViewController( - shippingMethods: [method1, method2], - selectedShippingMethod: method1, - currency: "usd", - theme: STPTheme.defaultTheme - ) - let viewToTest = stp_preparedAndSizedViewForSnapshotTest(from: shippingVC)! - STPSnapshotVerifyView(viewToTest, identifier: nil) - STPLocalizationUtils.overrideLanguage(to: nil) - } - - func testGerman() { - performSnapshotTest(forLanguage: "de") - } - - func testEnglish() { - performSnapshotTest(forLanguage: "en") - } - - func testSpanish() { - performSnapshotTest(forLanguage: "es") - } - - func testFrench() { - performSnapshotTest(forLanguage: "fr") - } - - func testItalian() { - performSnapshotTest(forLanguage: "it") - } - - func testJapanese() { - performSnapshotTest(forLanguage: "ja") - } - - func testDutch() { - performSnapshotTest(forLanguage: "nl") - } - - func testChinese() { - performSnapshotTest(forLanguage: "zh-Hans") - } -} diff --git a/Stripe/StripeiOSTests/STPSwiftFixtures.swift b/Stripe/StripeiOSTests/STPSwiftFixtures.swift index f522531e350..b3af99c5a6c 100644 --- a/Stripe/StripeiOSTests/STPSwiftFixtures.swift +++ b/Stripe/StripeiOSTests/STPSwiftFixtures.swift @@ -42,66 +42,3 @@ class MockEphemeralKeyProvider: NSObject, STPCustomerEphemeralKeyProvider { completion(STPFixtures.ephemeralKey().allResponseFields, nil) } } - -@objcMembers -@objc class Testing_StaticCustomerContext_Objc: Testing_StaticCustomerContext { - -} - -@objcMembers -class Testing_StaticCustomerContext: STPCustomerContext { - var customer: STPCustomer - var paymentMethods: [STPPaymentMethod] - convenience init() { - let customer = STPFixtures.customerWithSingleCardTokenSource() - let paymentMethods = [STPFixtures.paymentMethod()].compactMap { $0 } - self.init( - customer: customer, - paymentMethods: paymentMethods - ) - } - init( - customer: STPCustomer, - paymentMethods: [STPPaymentMethod] - ) { - self.customer = customer - self.paymentMethods = paymentMethods - super.init( - keyManager: STPEphemeralKeyManager( - keyProvider: MockEphemeralKeyProvider(), - apiVersion: "1", - performsEagerFetching: false - ), - apiClient: STPAPIClient.shared - ) - } - - override func retrieveCustomer(_ completion: STPCustomerCompletionBlock?) { - if let completion = completion { - completion(customer, nil) - } - } - - override func listPaymentMethodsForCustomer(completion: STPPaymentMethodsCompletionBlock?) { - if let completion = completion { - completion(paymentMethods, nil) - } - } - - var didAttach = false - override func attachPaymentMethod( - toCustomer paymentMethod: STPPaymentMethod, - completion: STPErrorBlock? - ) { - didAttach = true - if let completion = completion { - completion(nil) - } - } - - override func retrieveLastSelectedPaymentMethodIDForCustomer( - completion: @escaping (String?, Error?) -> Void - ) { - completion(nil, nil) - } -} diff --git a/Stripe/StripeiOSTests/UINavigationBar+StripeTest.m b/Stripe/StripeiOSTests/UINavigationBar+StripeTest.m deleted file mode 100644 index bc5a2bd8679..00000000000 --- a/Stripe/StripeiOSTests/UINavigationBar+StripeTest.m +++ /dev/null @@ -1,52 +0,0 @@ -// -// UINavigationBar+StripeTest.m -// Stripe -// -// Created by Brian Dorfman on 12/9/16. -// Copyright © 2016 Stripe, Inc. All rights reserved. -// - -#import -#import -@import Stripe; -#import "STPMocks.h" -@import StripePaymentsObjcTestUtils; - -@interface UINavigationBar_StripeTest : XCTestCase - -@end - -@implementation UINavigationBar_StripeTest - -- (STPPaymentOptionsViewController *)buildPaymentOptionsViewController { - id customerContext = [STPMocks staticCustomerContext]; - STPPaymentConfiguration *config = [STPPaymentConfiguration new]; - STPTheme *theme = [STPTheme defaultTheme]; - id delegate = OCMProtocolMock(@protocol(STPPaymentOptionsViewControllerDelegate)); - STPPaymentOptionsViewController *paymentOptionsVC = [[STPPaymentOptionsViewController alloc] initWithConfiguration:config - theme:theme - customerContext:customerContext - delegate:delegate]; - return paymentOptionsVC; -} - -- (void)testVCUsesNavigationBarColor { - STPPaymentOptionsViewController *paymentOptionsVC = [self buildPaymentOptionsViewController]; - STPTheme *navTheme = [STPTheme new]; - navTheme.accentColor = [UIColor purpleColor]; - - UINavigationController *navController = [[UINavigationController alloc] initWithRootViewController:paymentOptionsVC]; - navController.navigationBar.stp_theme = navTheme; - __unused UIView *view = paymentOptionsVC.view; - XCTAssertEqualObjects(paymentOptionsVC.navigationItem.leftBarButtonItem.tintColor, [UIColor purpleColor]); -} - -- (void)testVCDoesNotUseNavigationBarColor { - STPPaymentOptionsViewController *paymentOptionsVC = [self buildPaymentOptionsViewController]; - __unused UINavigationController *navController = [[UINavigationController alloc] initWithRootViewController:paymentOptionsVC]; - __unused UIView *view = paymentOptionsVC.view; - XCTAssertEqualObjects(paymentOptionsVC.navigationItem.leftBarButtonItem.tintColor, [STPTheme defaultTheme].accentColor); -} - - -@end