代码之家  ›  专栏  ›  技术社区  ›  Paul Metas

应用内购买和恢复按钮:单一产品-非消费品

  •  1
  • Paul Metas  · 技术社区  · 8 年前

    这个问题让我苦恼了好几天。

    我有一个简单的应用程序,可以显示横幅和间隙广告。

    我使用的是单视图应用程序,具有主视图控制器( ViewController.swift )并设置了另一个视图控制器( InAppViewController.swift )处理弹出页面:

    1. 允许用户进行应用内购买以删除所有广告(AdBanners和InterstitalAds);或
    2. 恢复购买。

    我的代码在运行时没有错误。

    应用内购买运行正常,但偶尔我会收到两次iTunes的登录请求。

    但我的恢复按钮和相关功能似乎是问题所在。

    我已经建立了许多沙盒测试帐户进行测试,一个没有购买该应用程序的新用户能够成功恢复购买。这不可能,所以我肯定做错了什么。

    这是我的代码:

    主视图控制器:

    //  ViewController.swift
    
    import UIKit
    import MessageUI
    import Social
    import iAd
    import StoreKit
    
    class ViewController: UIViewController, MFMailComposeViewControllerDelegate, MFMessageComposeViewControllerDelegate, ADBannerViewDelegate, ADInterstitialAdDelegate
    {
    
    let defaults = NSUserDefaults.standardUserDefaults()
    var product_id: NSString?;
    
    override func viewDidLoad() {
        product_id = "some.product.id";
        super.viewDidLoad()
    
         //Check if product is purchased
         if (defaults.boolForKey("purchased")){
            print("already purchased")
    
            // Hide or show banner ads is purchased/not purchased.
            // Advertising Banner:
            self.canDisplayBannerAds = false
            }
    
        else if (!defaults.boolForKey("stonerPurchased")){
            print("not yet purchased")
    
            // Advertising Banner:
            self.canDisplayBannerAds = true
            }
    

    这段代码似乎工作得很好。当应用程序加载时,它能够确定谁已经付费删除广告,谁还没有付费,广告横幅会适当显示。

    它在第二个视图控制器中( InAppPViewController.swift )我有问题。

    这是我的代码:

    第二视图控制器-InAppViewController.swift:

    //  InAppPViewController.swift
    
    import UIKit
    import StoreKit
    import iAd
    
    class InAppPViewController: UIViewController, SKProductsRequestDelegate, SKPaymentTransactionObserver {
    
    let defaults = NSUserDefaults.standardUserDefaults()
    var product_id: NSString?;
    
    
    @IBOutlet weak var unlockAction: UIButton!
    @IBOutlet var adBannerView: ADBannerView?
    
    override func viewDidLoad() {
         super.viewDidLoad()
    
        // Do any additional setup after loading the view.
        }
    
    override func didReceiveMemoryWarning() {
        super.didReceiveMemoryWarning()
        // Dispose of any resources that can be recreated.
    }
    
      @IBAction func restorePurchases(sender: UIButton) {
        SKPaymentQueue.defaultQueue().addTransactionObserver(self)
        SKPaymentQueue.defaultQueue().restoreCompletedTransactions()
    
    }
      func paymentQueueRestoreCompletedTransactionsFinished(queue: SKPaymentQueue) {
    
        print("Transactions Restored")
        let alert = UIAlertView(title: "Thank You", message: "Your purchase(s) were restored.", delegate: nil, cancelButtonTitle: "OK")
        alert.show()
    }
    
    
    @IBAction func unlockAction(sender: AnyObject) {
    
    product_id = "some.product.id";
    
        SKPaymentQueue.defaultQueue().addTransactionObserver(self)
    
        //Check if product is purchased
    
        if (defaults.boolForKey("purchased")){
    
        }
        else if (!defaults.boolForKey("stonerPurchased")){
            print("false")     
        }
    
    
        print("About to fetch the products");
        // We check that we are allowed to make the purchase.
    
        if (SKPaymentQueue.canMakePayments())
        {
            let productID:NSSet = NSSet(object: self.product_id!);
            let productsRequest:SKProductsRequest = SKProductsRequest(productIdentifiers: productID as! Set<String>);
            productsRequest.delegate = self;
            productsRequest.start();
            print("Fething Products");
        }else{
            print("can't make purchases");
        }
    
    }
    
    func buyProduct(product: SKProduct){
        print("Sending the Payment Request to Apple");
        let payment = SKPayment(product: product)
        SKPaymentQueue.defaultQueue().addPayment(payment);
    
    }
    //Delegate Methods for IAP
    
    func productsRequest (request: SKProductsRequest, didReceiveResponse response: SKProductsResponse) {
    
        let count : Int = response.products.count
        if (count>0) {
    
            let validProduct: SKProduct = response.products[0] as SKProduct
            if (validProduct.productIdentifier == self.product_id) {
                print(validProduct.localizedTitle)
                print(validProduct.localizedDescription)
                print(validProduct.price)
                buyProduct(validProduct);
            } else {
                print(validProduct.productIdentifier)
            }
        } else {
            print("nothing")
        }
    }
    
    
    func request(request: SKRequest, didFailWithError error: NSError) {
        print("Error Fetching product information");
    }
    
    func paymentQueue(queue: SKPaymentQueue, updatedTransactions transactions: [SKPaymentTransaction])   {
        print("Received Payment Transaction Response from Apple");
    
        for transaction:AnyObject in transactions {
            if let trans:SKPaymentTransaction = transaction as? SKPaymentTransaction{
                switch trans.transactionState {
    
                case .Purchased:
                    print("Product Purchased");
                    SKPaymentQueue.defaultQueue().finishTransaction(transaction as! SKPaymentTransaction)
                    defaults.setBool(true , forKey: "purchased")
                    break;
    
                case .Failed:
                    print("Purchased Failed");
                    SKPaymentQueue.defaultQueue().finishTransaction(transaction as! SKPaymentTransaction)
                    break;
    
                case .Restored:
                    print("Already Purchased");
                    SKPaymentQueue.defaultQueue().restoreCompletedTransactions()
    
    
                default:
                    break;
                }
            }
        }
    
    }
    

    }

    我哪里出错了?

    问题:

    1. 我的上述代码正确吗?
    2. 我应该修改什么?为什么?

    提前道歉,我对这个美妙的编码世界很陌生…但我热爱它的每一分钟!

    2 回复  |  直到 8 年前
        1
  •  2
  •   Done    8 年前
    1. SKPaymentQueue.defaultQueue().addTransactionObserver(self) 应该在 viewDidLoad() 不在还原功能中
    2. 你可以把 SKPaymentQueue.defaultQueue().restoreCompletedTransactions() 这样地 override func viewWillDisappear(animated: Bool) { SKPaymentQueue.defaultQueue().removeTransactionObserver(self) }
    3. 在沙盒中登录两次是正常的。

    希望我能帮上忙。

        2
  •  0
  •   Paul Metas    8 年前

    我已经修改了我的代码 InAppPViewController.swift 文件如下:

    //  InAppPViewController.swift
    
    import UIKit
    import StoreKit
    
    class InAppPViewController: UIViewController, SKProductsRequestDelegate, SKPaymentTransactionObserver {
    
    let defaults = NSUserDefaults.standardUserDefaults()
    var product_id: NSString?;
    
    override func viewDidLoad() {
        super.viewDidLoad()
    
        // Do any additional setup after loading the view.  
    }
    
    override func didReceiveMemoryWarning() {
        super.didReceiveMemoryWarning()
        // Dispose of any resources that can be recreated.
    }
    
    @IBAction func restorePurchases(sender: UIButton) {
        // Set up the observer
        SKPaymentQueue.defaultQueue().addTransactionObserver(self)
    
    
        //Check if user can make payments and then proceed to restore purchase
        if (SKPaymentQueue.canMakePayments()) {
            SKPaymentQueue.defaultQueue().restoreCompletedTransactions()
        }
    
    }
    
    @IBAction func unlockAction(sender: AnyObject) {
    
        product_id = "some.product.id";
        // Adding the observer
        SKPaymentQueue.defaultQueue().addTransactionObserver(self)
    
        //Check if product is purchased
        if (defaults.boolForKey("purchased")){
            print("User already purchased this")
            // Hide a view or show content depends on your requirement
        }
    
        else if (!defaults.boolForKey("Purchased")){
            print("User has not yet pur hased this") 
        }
    
    
        print("About to fetch the products");
    
        // Check if user can make payments and then proceed to make the purchase.
        if (SKPaymentQueue.canMakePayments())
        {
            let productID:NSSet = NSSet(object: self.product_id!);
            let productsRequest:SKProductsRequest = SKProductsRequest(productIdentifiers: productID as! Set<String>);
            productsRequest.delegate = self;
            productsRequest.start();
            print("User can make purchases and will fetch products from Apple Store now");
        }else{
            print("User can't make purchases");
        }
    
    }
    
    func buyProduct(product: SKProduct){
        print("Sending the Payment Request to Apple");
        let payment = SKPayment(product: product)
        SKPaymentQueue.defaultQueue().addPayment(payment);
    
    }
    
    func productsRequest (request: SKProductsRequest, didReceiveResponse response: SKProductsResponse) {
    
        let count : Int = response.products.count
        if (count>0) {
    
            let validProduct: SKProduct = response.products[0] as SKProduct
            if (validProduct.productIdentifier == self.product_id) {
                print(validProduct.localizedTitle)
                print(validProduct.localizedDescription)
                print(validProduct.price)
                buyProduct(validProduct);
            } else {
                print(validProduct.productIdentifier)
            }
        } else {
            print("nothing")
        }
    }
    
    
    func request(request: SKRequest, didFailWithError error: NSError) {
        print("Error Fetching product information");
    }
    
    // Allowing for all possible outcomes:
    func paymentQueue(queue: SKPaymentQueue, updatedTransactions transactions: [SKPaymentTransaction])   {
        print("Received Payment Transaction Response from Apple");
    
        for transaction:AnyObject in transactions {
            if let trans:SKPaymentTransaction = transaction as? SKPaymentTransaction {
                switch trans.transactionState {
    
                case .Purchased:
                    print("Product Purchased")
                    let alert = UIAlertView(title: "Thank You", message: "Thank you for your purchase!", delegate: nil, cancelButtonTitle: "OK")
                    alert.show();
                    SKPaymentQueue.defaultQueue().finishTransaction(transaction as! SKPaymentTransaction)
                    defaults.setBool(true , forKey: "purchased")
                    break;
    
                case .Failed:
                    print("Purchased Failed");
                    SKPaymentQueue.defaultQueue().finishTransaction(transaction as! SKPaymentTransaction)
                    break;
    
                case .Restored:
                    print("Already Purchased");
                    SKPaymentQueue.defaultQueue().restoreCompletedTransactions()
                    break;
    
                default:
                    break;
                }
            }
        }
    
    }
    

    }

    我离开了 ViewController.swift 文件。

    产品购买现在似乎奏效了。

    但是关于恢复购买,我可以在我的物理设备上运行代码,但是 不能 测试“恢复购买”功能。

    我被抓住了 以前的 恢复尚未解决并在系统中循环的购买。我无法手动清除我的SKPaymentsQueue。因此,我的代码拒绝接受任何新的恢复购买请求。