【问题标题】:In-App Purchase & Restore Button : Single Product - Non-Consumable应用内购买和恢复按钮:单一产品 - 非消耗品
【发布时间】:2016-02-10 12:57:19
【问题描述】:

这个问题困扰了我好几天。

我有一个简单的应用程序,它显示横幅和插页式广告。

我正在使用单视图应用程序,拥有主视图控制器 (ViewController.swift) 并设置了另一个视图控制器 (InAppViewController.swift) 来处理以下弹出页面:

  1. 允许用户进行应用内购买以移除所有广告(广告横幅和插页式广告);或
  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. 我应该修改什么以及为什么?

提前道歉,我是这个美妙的编码世界的新手......但我热爱它的每一分钟!

【问题讨论】:

  • 不要将您的答案编辑到您的问题中。将其发布为答案。
  • @Daniel Storm 但是,如果我不确定我的答案是否正确,并且我发布了我认为是答案的内容,那么我将有效地关闭我的问题作为回答?而且我不会收到更好的答案?
  • 您可以发布您的答案并且不接受它,这样其他用户仍然可以发布答案。好吧,如果您接受您的答案,他们也可以发布,但它确实可以发送您找到解决方案的消息。所以,是的:您可以从问题中删除它并将其作为答案发布。如果有人发布了更好的,您可以随时删除您的。

标签: ios swift in-app-purchase iad restore


【解决方案1】:
  1. SKPaymentQueue.defaultQueue().addTransactionObserver(self) 应该在 viewDidLoad() 中而不是在恢复函数中
  2. 你可以这样写SKPaymentQueue.defaultQueue().restoreCompletedTransactions() override func viewWillDisappear(animated: Bool) { SKPaymentQueue.defaultQueue().removeTransactionObserver(self) }
  3. 在沙盒中登录两次是正常的。

希望我能帮忙。

【讨论】:

    【解决方案2】:

    我已将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 文件原样保留。

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

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

    我遇到了previous恢复购买仍然未解决并在系统中循环。我无法手动清除我的 SKPaymentsQueue。因此,我的代码不再接受新的恢复购买请求。

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多