Files
sodalive-ios/SodaLive/Sources/MyPage/Can/Payment/CanPgPaymentViewModel.swift

281 lines
12 KiB
Swift

//
// CanPgPaymentViewModel.swift
// SodaLive
//
// Created by klaus on 2023/08/11.
//
import Foundation
import Combine
import Bootpay
enum PaymentMethod: String {
case unified = "통합 결제"
case phone = "휴대폰"
case kakaopay = "카카오페이"
}
final class CanPgPaymentViewModel: ObservableObject {
private let repository = CanRepository()
private var subscription = Set<AnyCancellable>()
@Published var isTermsAgree = false
@Published var errorMessage = ""
@Published var isShowPopup = false
@Published var isLoading = false
@Published var isShowPaymentView = false
@Published var isShowPayversePaymentView = false
@Published var paymentMethod: PaymentMethod? = nil
let payload = Payload()
var payversePayloadJson: String = ""
var canResponse: GetCanResponse? = nil
var refresh: (() -> Void)? = nil
var afterCompletionToGoBack: Bool? = nil
func chargeCan(canId: Int, onSuccess: @escaping () -> Void) {
isLoading = true
repository.pgChargeCan(canId: canId)
.sink { result in
switch result {
case .finished:
DEBUG_LOG("finish")
case .failure(let error):
ERROR_LOG(error.localizedDescription)
}
} receiveValue: { [unowned self] response in
self.isLoading = false
let responseData = response.data
do {
let jsonDecoder = JSONDecoder()
let decoded = try jsonDecoder.decode(ApiResponse<CanChargeResponse>.self, from: responseData)
if let data = decoded.data, decoded.success {
payload.applicationId = BOOTPAY_APP_HECTO_ID
payload.pg = paymentMethod == .kakaopay ? "카카오" : "세틀뱅크"
payload.orderId = "\(data.chargeId)"
payload.method = paymentMethod!.rawValue
var username = UserDefaults.string(forKey: .nickname)
if username.count > 10 {
username = "\(username.prefix(6))..."
}
let bootUser = BootUser()
bootUser.userId = "\(UserDefaults.int(forKey: .userId))"
bootUser.username = UserDefaults.string(forKey: .nickname)
payload.user = bootUser
onSuccess()
} else {
if let message = decoded.message {
self.errorMessage = message
} else {
self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
}
self.isShowPopup = true
}
} catch {
self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
self.isShowPopup = true
}
}
.store(in: &subscription)
}
func verifyPayment(_ data: [String: Any], onSuccess: @escaping () -> Void) {
isLoading = true
let _data = data["data"] as? [String: Any]
if let data = _data {
let receiptId = data["receipt_id"] as! String
let orderId = data["order_id"] as! String
repository.pgVerifyHecto(receiptId: receiptId, orderId: orderId)
.sink { result in
switch result {
case .finished:
DEBUG_LOG("finish")
case .failure(let error):
ERROR_LOG(error.localizedDescription)
}
} receiveValue: { [unowned self] response in
self.isLoading = false
let responseData = response.data
do {
let jsonDecoder = JSONDecoder()
let decoded = try jsonDecoder.decode(ApiResponseWithoutData.self, from: responseData)
if decoded.success {
self.errorMessage = "캔이 충전되었습니다"
self.isShowPopup = true
onSuccess()
} else {
if let message = decoded.message {
self.errorMessage = message
} else {
self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
}
self.isShowPopup = true
}
} catch {
self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
self.isShowPopup = true
}
}
.store(in: &subscription)
} else {
isLoading = false
self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
isShowPopup = true
}
}
func payverseChargeCan(canId: Int) {
isLoading = true
repository.payverseChargeCan(canId: canId)
.sink { result in
switch result {
case .finished:
DEBUG_LOG("finish")
case .failure(let error):
ERROR_LOG(error.localizedDescription)
}
} receiveValue: { [unowned self] response in
self.isLoading = false
let responseData = response.data
do {
let jsonDecoder = JSONDecoder()
let decoded = try jsonDecoder.decode(ApiResponse<PayverseChargeResponse>.self, from: responseData)
if let data = decoded.data, decoded.success {
if let payloadJson = data.payloadJson.data(using: .utf8) {
var obj = try JSONSerialization.jsonObject(with: payloadJson, options: []) as? [String: Any] ?? [:]
obj["returnUrl"] = "\(APPSCHEME)://payverse/result"
obj["webhookUrl"] = "\(BASE_URL)/charge/payverse/webhook"
obj["appScheme"] = APPSCHEME
let merged = try JSONSerialization.data(withJSONObject: obj, options: [])
self.payversePayloadJson = String(data: merged, encoding: .utf8)!
self.isShowPayversePaymentView = true
} else {
self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
self.isShowPopup = true
return
}
} else {
if let message = decoded.message {
self.errorMessage = message
} else {
self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
}
self.isShowPopup = true
}
} catch {
self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
self.isShowPopup = true
}
}
.store(in: &subscription)
}
func handleVerifyOpenURL(_ url: URL) {
guard let comps = URLComponents(url: url, resolvingAgainstBaseURL: false) else {
self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
self.isShowPopup = true
return
}
let q = comps.queryItems?.reduce(into: [String:String]()) { $0[$1.name] = $1.value } ?? [:]
let tid = q["tid"] ?? q["tx_id"]
let orderId = q["orderId"]
let resultStatus = q["resultStatus"]
if resultStatus == "FAILED" ||
resultStatus == "DECLINE" ||
orderId == nil || orderId?.trimmingCharacters(in: .whitespacesAndNewlines).isEmpty == true ||
tid == nil || tid?.trimmingCharacters(in: .whitespacesAndNewlines).isEmpty == true {
self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
self.isShowPopup = true
DispatchQueue.main.asyncAfter(deadline: .now() + 1) {
AppState.shared.back()
}
return
} else {
payverseVerify(transactionId: tid!, orderId: orderId!)
}
}
func payverseVerify(transactionId: String, orderId: String) {
isLoading = true
repository.payverseVerify(transactionId: transactionId, orderId: orderId)
.sink { result in
switch result {
case .finished:
DEBUG_LOG("finish")
case .failure(let error):
ERROR_LOG(error.localizedDescription)
}
} receiveValue: { [unowned self] response in
self.isLoading = false
let responseData = response.data
do {
let jsonDecoder = JSONDecoder()
let decoded = try jsonDecoder.decode(ApiResponseWithoutData.self, from: responseData)
if decoded.success {
if let canResponse = self.canResponse {
let can = UserDefaults.int(forKey: .can)
UserDefaults.set(can + canResponse.can + canResponse.rewardCan, forKey: .can)
if let refresh = refresh {
refresh()
}
DispatchQueue.main.asyncAfter(deadline: .now() + 1) {
if let afterCompletionToGoBack = self.afterCompletionToGoBack, afterCompletionToGoBack {
AppState.shared.back()
AppState.shared.back()
} else {
if let refresh = self.refresh {
AppState.shared.setAppStep(step: .canStatus(refresh: refresh))
} else {
AppState.shared.setAppStep(step: .canStatus(refresh: {}))
}
}
}
} else {
AppState.shared.setAppStep(step: .canStatus(refresh: {}))
}
} else {
if let message = decoded.message {
self.errorMessage = message
} else {
self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
}
self.isShowPopup = true
}
} catch {
self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
self.isShowPopup = true
}
}
.store(in: &subscription)
}
}