-
Notifications
You must be signed in to change notification settings - Fork 129
/
Copy pathCardComponentAdvancedFlowExample.swift
217 lines (174 loc) · 7.14 KB
/
CardComponentAdvancedFlowExample.swift
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
//
// Copyright (c) 2023 Adyen N.V.
//
// This file is open source and available under the MIT license. See the LICENSE file for more info.
//
import Adyen
import AdyenActions
import AdyenCard
import AdyenComponents
import AdyenDropIn
import PassKit
import UIKit
internal final class CardComponentAdvancedFlowExample: InitialDataAdvancedFlowProtocol {
// MARK: - Properties
internal var cardComponent: PresentableComponent?
internal weak var presenter: PresenterExampleProtocol?
internal lazy var apiClient = ApiClientHelper.generateApiClient()
internal lazy var context: AdyenContext = generateContext()
// MARK: - Action Handling
private lazy var adyenActionComponent: AdyenActionComponent = {
let handler = AdyenActionComponent(context: context)
handler.configuration.threeDS.delegateAuthentication = ConfigurationConstants.delegatedAuthenticationConfigurations
handler.configuration.threeDS.requestorAppURL = ConfigurationConstants.returnUrl
handler.delegate = self
handler.presentationDelegate = self
return handler
}()
// MARK: - Initializers
internal init() {}
internal func start() {
presenter?.showLoadingIndicator()
requestPaymentMethods(order: nil) { [weak self] result in
guard let self else { return }
self.presenter?.hideLoadingIndicator()
switch result {
case let .success(paymentMethods):
self.presentComponent(with: paymentMethods)
case let .failure(error):
self.presentAlert(with: error)
}
}
}
// MARK: - Presentation
private func presentComponent(with paymentMethods: PaymentMethods) {
do {
let component = try cardComponent(from: paymentMethods)
let componentViewController = viewController(for: component)
presenter?.present(viewController: componentViewController, completion: nil)
cardComponent = component
} catch {
self.presentAlert(with: error)
}
}
private func cardComponent(from paymentMethods: PaymentMethods) throws -> CardComponent {
guard let paymentMethod = paymentMethods.paymentMethod(ofType: CardPaymentMethod.self) else {
throw IntegrationError.paymentMethodNotAvailable(paymentMethod: CardPaymentMethod.self)
}
let component = CardComponent(
paymentMethod: paymentMethod,
context: context,
configuration: ConfigurationConstants.current.cardConfiguration
)
component.cardComponentDelegate = self
component.delegate = self
return component
}
private func viewController(for component: PresentableComponent) -> UIViewController {
guard component.requiresModalPresentation else {
return component.viewController
}
let navigation = UINavigationController(rootViewController: component.viewController)
component.viewController.navigationItem.leftBarButtonItem = .init(
barButtonSystemItem: .cancel,
target: self,
action: #selector(cancelPressed)
)
return navigation
}
@objc private func cancelPressed() {
cardComponent?.cancelIfNeeded()
presenter?.dismiss(completion: nil)
}
// MARK: - Payment response handling
private func paymentResponseHandler(result: Result<PaymentsResponse, Error>) {
switch result {
case let .success(response):
if let action = response.action {
adyenActionComponent.handle(action)
} else {
finish(with: response)
}
case let .failure(error):
finish(with: error)
}
}
private func finish(with result: PaymentsResponse) {
let success = result.isAccepted
let message = "\(result.resultCode.rawValue) \(result.amount?.formatted ?? "")"
finalize(success, message)
}
private func finish(with error: Error) {
let message: String
if let componentError = (error as? ComponentError), componentError == ComponentError.cancelled {
message = "Cancelled"
} else {
message = error.localizedDescription
}
finalize(false, message)
}
private func finalize(_ success: Bool, _ message: String) {
cardComponent?.finalizeIfNeeded(with: success) { [weak self] in
guard let self else { return }
self.dismissAndShowAlert(success, message)
}
}
private func presentAlert(with error: Error, retryHandler: (() -> Void)? = nil) {
presenter?.presentAlert(with: error, retryHandler: retryHandler)
}
private func dismissAndShowAlert(_ success: Bool, _ message: String) {
presenter?.dismiss {
// Payment is processed. Add your code here.
let title = success ? "Success" : "Error"
self.presenter?.presentAlert(withTitle: title, message: message)
}
}
}
extension CardComponentAdvancedFlowExample: CardComponentDelegate {
func didSubmit(lastFour: String, finalBIN: String, component: CardComponent) {
print("Card used: **** **** **** \(lastFour)")
print("Final BIN: \(finalBIN)")
}
internal func didChangeBIN(_ value: String, component: CardComponent) {
print("Current BIN: \(value)")
}
internal func didChangeCardBrand(_ value: [CardBrand]?, component: CardComponent) {
print("Current card type: \((value ?? []).reduce("") { "\($0), \($1)" })")
}
}
extension CardComponentAdvancedFlowExample: PaymentComponentDelegate {
internal func didSubmit(_ data: PaymentComponentData, from component: PaymentComponent) {
let request = PaymentsRequest(data: data)
apiClient.perform(request) { [weak self] result in
self?.paymentResponseHandler(result: result)
}
}
internal func didFail(with error: Error, from component: PaymentComponent) {
finish(with: error)
}
}
extension CardComponentAdvancedFlowExample: ActionComponentDelegate {
internal func didFail(with error: Error, from component: ActionComponent) {
finish(with: error)
}
internal func didComplete(from component: ActionComponent) {
finish(with: .received)
}
internal func didProvide(_ data: ActionComponentData, from component: ActionComponent) {
(component as? PresentableComponent)?.viewController.view.isUserInteractionEnabled = false
let request = PaymentDetailsRequest(
details: data.details,
paymentData: data.paymentData,
merchantAccount: ConfigurationConstants.current.merchantAccount
)
apiClient.perform(request) { [weak self] result in
self?.paymentResponseHandler(result: result)
}
}
}
extension CardComponentAdvancedFlowExample: PresentationDelegate {
internal func present(component: PresentableComponent) {
let componentViewController = viewController(for: component)
presenter?.present(viewController: componentViewController, completion: nil)
}
}