티스토리 뷰

728x90
반응형

 

 

목록

  1. 앱 스토어(App Store)에 제품 등록 하기
  2. 앱 스토어(App Store)에서 제품 정보 및 목록 가져오기
  3. 앱 스토어(App Store)에 제품 결제 요청 및 응답  처리하기
  4. In App Purchase API

 

 

앱 스토어(App Store)에서 제품 목록 및 정보 가져오기

//
//  AppStore.swift
//  DoctorSL
//
//  Created by 장정훈 on 2021/10/30.
//

import StoreKit

class AppStore: NSObject, ObservableObject {    
    @Published var inAppProducts = [InAppProduct]()
    
    // MARK: 제품 ID(App Store Connect -> App Store -> 구독)
    private let allProductIdentifiers = Set([
        "net.doctorsl.yearly",
        "net.doctorsl.monthly"
    ])
    
    private var productsRequest: SKProductsRequest?
    private var fetchedProducts = [SKProduct]()
    private var fetchCompletionHandler: (([SKProduct]) -> Void)?
    
    override init() {
        super.init()
        
        // MARK: 1) 앱 스토어(App Store)에 제품 정보를 가져옵니다.
        fetchProducts() { products in // "fetchProducts" Closer 구현
            self.inAppProducts = products.map { InAppProduct(product: $0) }
        }
    }
    
    // MARK: 1.1) 앱 스토어(App Store)에 제품 정보를 요청합니다.
    private func fetchProducts(_ completion: @escaping ([SKProduct]) -> Void) { // "fetchProducts" @escaping Closer 선언
        guard self.productsRequest == nil else { return }
        
        // 클로저(Closer)를 외부 변수에 저장합니다.
        // 클로저(Closer)를 명시적으로 호출 하지 않고 외부 변수에 저장 했기 때문에
        // 클로저(Closer)의 구현부는 실행되지 않습니다. 하지만
        // "fetchProducts" 함수는 실행이 됩니다.(클로저(Closer)의 구현부만 실행되지 않습니다.)
        self.fetchCompletionHandler = completion
        
        // SKProductsRequest 객체 인스턴스를 생성합니다.
        self.productsRequest = SKProductsRequest(productIdentifiers: allProductIdentifiers)
        
        // 요청에 대한 응답을 처리할 대리자(delegate)를 자신으로 설정합니다.
        // 요청에 대한 응답은 SKProductsRequestDelegate 대리자(delegate)가 처리합니다.
        productsRequest?.delegate = self
        
        // 앱 스토어(App Store)에 설정된 제품 목록 및 정보를 요청합니다.
        productsRequest?.start()
    }
}

// MARK: 1.2) 제품 정보 요청에 대한 앱 스토어(App Store)의 응답을 처리합니다.
extension AppStore: SKProductsRequestDelegate {    
    func productsRequest(_ request: SKProductsRequest, didReceive response: SKProductsResponse) {
        let loadedProducts = response.products
        let invalidProducts = response.invalidProductIdentifiers
        
        guard !loadedProducts.isEmpty else {
            if !invalidProducts.isEmpty {
                fatalError("Invalid products found: \(invalidProducts)")
            }
            
            productsRequest = nil
            return
        }
        
        self.fetchedProducts = loadedProducts
        
        DispatchQueue.main.async {
            // 클로저(Closer)를 명시적으로 호출 합니다.
            // 클로저(Closer)의 구현부가 실행됩니다.
            self.fetchCompletionHandler?(loadedProducts)
            
            // 클로저(Closer)를 초기화 합니다.
            self.fetchCompletionHandler = nil
            self.productsRequest = nil
        }
    }
}

 

 

 

728x90
반응형